151 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			151 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /****************************************************************************
 | |
| **
 | |
| ** Copyright (C) 2015 The Qt Company Ltd.
 | |
| ** Contact: http://www.qt.io/licensing/
 | |
| **
 | |
| ** This file is part of the examples of the Qt Toolkit.
 | |
| **
 | |
| ** $QT_BEGIN_LICENSE:BSD$
 | |
| ** You may use this file under the terms of the BSD license as follows:
 | |
| **
 | |
| ** "Redistribution and use in source and binary forms, with or without
 | |
| ** modification, are permitted provided that the following conditions are
 | |
| ** met:
 | |
| **   * Redistributions of source code must retain the above copyright
 | |
| **     notice, this list of conditions and the following disclaimer.
 | |
| **   * Redistributions in binary form must reproduce the above copyright
 | |
| **     notice, this list of conditions and the following disclaimer in
 | |
| **     the documentation and/or other materials provided with the
 | |
| **     distribution.
 | |
| **   * Neither the name of The Qt Company Ltd nor the names of its
 | |
| **     contributors may be used to endorse or promote products derived
 | |
| **     from this software without specific prior written permission.
 | |
| **
 | |
| **
 | |
| ** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 | |
| ** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 | |
| ** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
 | |
| ** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 | |
| ** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 | |
| ** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 | |
| ** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 | |
| ** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 | |
| ** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 | |
| ** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 | |
| ** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
 | |
| **
 | |
| ** $QT_END_LICENSE$
 | |
| **
 | |
| ****************************************************************************/
 | |
| 
 | |
| #include <qendian.h>
 | |
| #include <QVector>
 | |
| #include <QDebug>
 | |
| #include "WaveFile.h"
 | |
| 
 | |
| struct chunk
 | |
| {
 | |
|     char        id[4];
 | |
|     quint32     size;
 | |
| };
 | |
| 
 | |
| struct RIFFHeader
 | |
| {
 | |
|     chunk       descriptor;     // "RIFF"
 | |
|     char        type[4];        // "WAVE"
 | |
| };
 | |
| 
 | |
| struct WAVEHeader
 | |
| {
 | |
|     chunk       descriptor;
 | |
|     quint16     audioFormat;
 | |
|     quint16     numChannels;
 | |
|     quint32     sampleRate;
 | |
|     quint32     byteRate;
 | |
|     quint16     blockAlign;
 | |
|     quint16     bitsPerSample;
 | |
| };
 | |
| 
 | |
| struct DATAHeader
 | |
| {
 | |
|     chunk       descriptor;
 | |
| };
 | |
| 
 | |
| struct CombinedHeader
 | |
| {
 | |
|     RIFFHeader  riff;
 | |
|     WAVEHeader  wave;
 | |
| };
 | |
| 
 | |
| WaveFile::WaveFile(QObject *parent)
 | |
|     : QFile(parent)
 | |
|     , m_headerLength(0)
 | |
| {
 | |
| 
 | |
| }
 | |
| 
 | |
| bool WaveFile::open(const QString &fileName)
 | |
| {
 | |
|     close();
 | |
|     setFileName(fileName);
 | |
|     return QFile::open(QIODevice::ReadOnly) && readHeader();
 | |
| }
 | |
| 
 | |
| const QAudioFormat &WaveFile::fileFormat() const
 | |
| {
 | |
|     return m_fileFormat;
 | |
| }
 | |
| 
 | |
| qint64 WaveFile::headerLength() const
 | |
| {
 | |
| return m_headerLength;
 | |
| }
 | |
| 
 | |
| bool WaveFile::readHeader()
 | |
| {
 | |
|     seek(0);
 | |
|     CombinedHeader header;
 | |
|     bool result = read(reinterpret_cast<char *>(&header), sizeof(CombinedHeader)) == sizeof(CombinedHeader);
 | |
|     if (result) {
 | |
|         if ((memcmp(&header.riff.descriptor.id, "RIFF", 4) == 0
 | |
|             || memcmp(&header.riff.descriptor.id, "RIFX", 4) == 0)
 | |
|             && memcmp(&header.riff.type, "WAVE", 4) == 0
 | |
|             && memcmp(&header.wave.descriptor.id, "fmt ", 4) == 0
 | |
|             && (header.wave.audioFormat == 1 || header.wave.audioFormat == 0)) {
 | |
| 
 | |
|             // Read off remaining header information
 | |
|             DATAHeader dataHeader;
 | |
| 
 | |
|             if (qFromLittleEndian<quint32>(header.wave.descriptor.size) > sizeof(WAVEHeader)) {
 | |
|                 // Extended data available
 | |
|                 quint16 extraFormatBytes;
 | |
|                 if (peek((char*)&extraFormatBytes, sizeof(quint16)) != sizeof(quint16))
 | |
|                     return false;
 | |
|                 const qint64 throwAwayBytes = sizeof(quint16) + qFromLittleEndian<quint16>(extraFormatBytes);
 | |
|                 if (read(throwAwayBytes).size() != throwAwayBytes)
 | |
|                     return false;
 | |
|             }
 | |
| 
 | |
|             if (read((char*)&dataHeader, sizeof(DATAHeader)) != sizeof(DATAHeader))
 | |
|                 return false;
 | |
| 
 | |
|             // Establish format
 | |
|             if (memcmp(&header.riff.descriptor.id, "RIFF", 4) == 0)
 | |
|                 m_fileFormat.setByteOrder(QAudioFormat::LittleEndian);
 | |
|             else
 | |
|                 m_fileFormat.setByteOrder(QAudioFormat::BigEndian);
 | |
| 
 | |
|             int bps = qFromLittleEndian<quint16>(header.wave.bitsPerSample);
 | |
|             m_fileFormat.setChannelCount(qFromLittleEndian<quint16>(header.wave.numChannels));
 | |
|             m_fileFormat.setCodec("audio/pcm");
 | |
|             m_fileFormat.setSampleRate(qFromLittleEndian<quint32>(header.wave.sampleRate));
 | |
|             m_fileFormat.setSampleSize(qFromLittleEndian<quint16>(header.wave.bitsPerSample));
 | |
|             m_fileFormat.setSampleType(bps == 8 ? QAudioFormat::UnSignedInt : QAudioFormat::SignedInt);
 | |
|         } else {
 | |
|             result = false;
 | |
|         }
 | |
|     }
 | |
|     m_headerLength = pos();
 | |
|     return result;
 | |
| }
 | 
