/*************************************************************************** * Copyright (C) 2007-2018 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 #include #ifdef Q_OS_WIN #include #define ENABLE_SNDFILE_WINDOWS_PROTOTYPES 1 #endif #include #include "decoder_sndfile.h" #include "decodersndfilefactory.h" #define WAVE_FORMAT_PCM 0x0001 #define WAVE_FORMAT_ADPCM 0x0002 #define WAVE_FORMAT_IEEE_FLOAT 0x0003 #define WAVE_FORMAT_ALAW 0x0006 #define WAVE_FORMAT_MULAW 0x0007 // DecoderSndFileFactory bool DecoderSndFileFactory::canDecode(QIODevice *input) const { char buf[36] = {0}; if(input->peek(buf, sizeof(buf)) != sizeof(buf)) return false; if(!memcmp(buf + 8, "WAVE", 4) && (!memcmp(buf, "RIFF", 4) || !memcmp(buf, "RIFX", 4))) { quint16 subformat = 0; if(!memcmp(buf + 12, "fmt ", 4)) { subformat = (quint16(buf[21]) << 8) + buf[20]; } else if(!input->isSequential()) { input->seek(12); //skip "JUNK" and "bext" chunks while(!input->atEnd()) { if(input->peek(buf, sizeof(buf)) != sizeof(buf)) return false; if(!memcmp(buf, "fmt ", 4)) { subformat = (quint16(buf[9]) << 8) + buf[8]; break; } else if(!memcmp(buf, "JUNK", 4) || !memcmp(buf, "bext", 4)) { size_t size = buf[4] + (buf[5] << 8) + (buf[6] << 16) + (buf[7] << 24); if(!input->seek(input->pos() + size + 8)) break; } else { break; } } input->seek(0); } switch (subformat) { case WAVE_FORMAT_PCM: case WAVE_FORMAT_ADPCM: case WAVE_FORMAT_IEEE_FLOAT: case WAVE_FORMAT_ALAW: case WAVE_FORMAT_MULAW: return true; default: return false; } } else if(!memcmp(buf, "FORM", 4)) { if(!memcmp(buf + 8, "AIFF", 4)) return true; if(!memcmp(buf + 8, "8SVX", 4)) return true; } else if(!memcmp(buf, ".snd", 4) || !memcmp(buf, "dns.", 4)) return true; else if(!memcmp(buf, "fap ", 4) || !memcmp(buf, " paf", 4)) return true; else if(!memcmp(buf, "NIST", 4)) return true; else if(!memcmp(buf, "Crea", 4) && !memcmp(buf + 4, "tive", 4)) return true; else if(!memcmp(buf, "riff", 4)) return true; return false; } DecoderProperties DecoderSndFileFactory::properties() const { DecoderProperties properties; properties.name = tr("Sndfile Plugin"); properties.filters << "*.wav" << "*.au" << "*.snd" << "*.aif" << "*.aiff" << "*.8svx"; properties.filters << "*.sph" << "*.sf" << "*.voc" << "*.w64"; properties.description = tr("PCM Files"); //properties.contentType = ""; properties.shortName = "sndfile"; properties.hasAbout = true; properties.hasSettings = false; properties.noInput = false; return properties; } Decoder *DecoderSndFileFactory::create(const QString &, QIODevice *input) { return new DecoderSndFile(input); } QList DecoderSndFileFactory::createPlayList(const QString &path, TrackInfo::Parts parts, QStringList *) { TrackInfo *info = new TrackInfo(path); if(parts == TrackInfo::NoParts) return QList() << info; SF_INFO snd_info; SNDFILE *sndfile = 0; memset(&snd_info, 0, sizeof(snd_info)); snd_info.format = 0; #ifdef Q_OS_WIN sndfile = sf_wchar_open(reinterpret_cast(path.utf16()), SFM_READ, &snd_info); #else sndfile = sf_open(path.toLocal8Bit().constData(), SFM_READ, &snd_info); #endif if(!sndfile) { delete info; return QList(); } if(parts & TrackInfo::MetaData) { const char *title = sf_get_string(sndfile, SF_STR_TITLE); info->setValue(Qmmp::TITLE, title ? QString::fromUtf8(title) : QString()); const char *date = sf_get_string(sndfile, SF_STR_DATE); info->setValue(Qmmp::YEAR, date ? QString::fromUtf8(date) : QString()); const char *album = sf_get_string(sndfile, SF_STR_ALBUM); info->setValue(Qmmp::ALBUM, album ? QString::fromUtf8(album) : QString()); const char *track = sf_get_string(sndfile, SF_STR_TRACKNUMBER); info->setValue(Qmmp::TRACK, track ? QString::fromUtf8(track) : QString()); const char *artist = sf_get_string(sndfile, SF_STR_ARTIST); info->setValue(Qmmp::ARTIST, artist ? QString::fromUtf8(artist) : QString()); const char *comment = sf_get_string(sndfile, SF_STR_COMMENT); info->setValue(Qmmp::COMMENT, comment ? QString::fromUtf8(comment) : QString()); const char *genre = sf_get_string(sndfile, SF_STR_GENRE); info->setValue(Qmmp::COMMENT, genre ? QString::fromUtf8(genre) : QString()); } if(parts & TrackInfo::Properties) { info->setValue(Qmmp::BITRATE, QFileInfo(path).size() * 8000.0 / info->duration() + 0.5); info->setValue(Qmmp::SAMPLERATE, snd_info.samplerate); info->setValue(Qmmp::CHANNELS, snd_info.channels); switch(snd_info.format & SF_FORMAT_SUBMASK) { case SF_FORMAT_PCM_S8: case SF_FORMAT_PCM_U8: info->setValue(Qmmp::BITS_PER_SAMPLE, 8); break; case SF_FORMAT_PCM_16: info->setValue(Qmmp::BITS_PER_SAMPLE, 16); break; case SF_FORMAT_PCM_24: info->setValue(Qmmp::BITS_PER_SAMPLE, 24); break; case SF_FORMAT_PCM_32: case SF_FORMAT_FLOAT: info->setValue(Qmmp::BITS_PER_SAMPLE, 32); break; case SF_FORMAT_DOUBLE: info->setValue(Qmmp::BITS_PER_SAMPLE, 64); break; } SF_FORMAT_INFO format_info; memset(&format_info, 0, sizeof(format_info)); format_info.format = (snd_info.format & SF_FORMAT_TYPEMASK); sf_command(0, SFC_GET_FORMAT_INFO, &format_info, sizeof(format_info)); info->setValue(Qmmp::FORMAT_NAME, QString::fromLatin1(format_info.name)); info->setDuration(int(snd_info.frames * 1000 / snd_info.samplerate)); } sf_close(sndfile); return QList() << info; } MetaDataModel* DecoderSndFileFactory::createMetaDataModel(const QString&, QObject *) { return 0; } void DecoderSndFileFactory::showSettings(QWidget *) {} void DecoderSndFileFactory::showAbout(QWidget *parent) { char version [128] = { 0 }; sf_command (NULL, SFC_GET_LIB_VERSION, version, sizeof (version)) ; QMessageBox::about (parent, tr("About Sndfile Audio Plugin"), tr("Qmmp Sndfile Audio Plugin")+"\n"+ tr("Compiled against")+" "+QString(version)+"\n" + tr("Written by: Ilya Kotov ")); } QString DecoderSndFileFactory::translation() const { return QLatin1String(":/sndfile_plugin_"); }