/* This file is part of the KDE project. Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). This library is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 2.1 or 3 of the License. This library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this library. If not, see . */ #include "iodevicereader.h" #include "qasyncreader.h" #include "mediagraph.h" #include #include #include QT_BEGIN_NAMESPACE #ifndef QT_NO_PHONON_ABSTRACTMEDIASTREAM namespace Phonon { namespace DS9 { //these mediatypes define a stream, its type will be autodetected by DirectShow static QVector getMediaTypes() { AM_MEDIA_TYPE mt; mt.majortype = MEDIATYPE_Stream; mt.bFixedSizeSamples = TRUE; mt.bTemporalCompression = FALSE; mt.lSampleSize = 1; mt.formattype = GUID_NULL; mt.pUnk = 0; mt.cbFormat = 0; mt.pbFormat = 0; QVector ret; //normal auto-detect stream mt.subtype = MEDIASUBTYPE_NULL; ret << mt; //AVI stream mt.subtype = MEDIASUBTYPE_Avi; ret << mt; //WAVE stream mt.subtype = MEDIASUBTYPE_WAVE; ret << mt; return ret; } class StreamReader : public QAsyncReader, public Phonon::StreamInterface { public: StreamReader(QBaseFilter *parent, const Phonon::MediaSource &source, const MediaGraph *mg) : QAsyncReader(parent, getMediaTypes()), m_seekable(false), m_pos(0), m_size(-1), m_mediaGraph(mg) { connectToSource(source); } //for Phonon::StreamInterface void writeData(const QByteArray &data) { QWriteLocker locker(&m_lock); m_pos += data.size(); m_buffer += data; } void endOfData() { } void setStreamSize(qint64 newSize) { QWriteLocker locker(&m_lock); m_size = newSize; } qint64 streamSize() const { QReadLocker locker(&m_lock); return m_size; } void setStreamSeekable(bool s) { QWriteLocker locker(&m_lock); m_seekable = s; } bool streamSeekable() const { QReadLocker locker(&m_lock); return m_seekable; } void setCurrentPos(qint64 pos) { QWriteLocker locker(&m_lock); m_pos = pos; seekStream(pos); m_buffer.clear(); } qint64 currentPos() const { QReadLocker locker(&m_lock); return m_pos; } int currentBufferSize() const { QReadLocker locker(&m_lock); return m_buffer.size(); } //virtual pure members //implementation from IAsyncReader STDMETHODIMP Length(LONGLONG *total, LONGLONG *available) { QReadLocker locker(&m_lock); if (total) { *total = m_size; } if (available) { *available = m_size; } return S_OK; } HRESULT read(LONGLONG pos, LONG length, BYTE *buffer, LONG *actual) { QMutexLocker locker(&m_mutexRead); if (m_mediaGraph->isStopping()) { return VFW_E_WRONG_STATE; } if(streamSize() != 1 && pos + length > streamSize()) { //it tries to read outside of the boundaries return E_FAIL; } if (currentPos() - currentBufferSize() != pos) { if (!streamSeekable()) { return S_FALSE; } setCurrentPos(pos); } int oldSize = currentBufferSize(); while (currentBufferSize() < int(length)) { needData(); if (m_mediaGraph->isStopping()) { return VFW_E_WRONG_STATE; } if (oldSize == currentBufferSize()) { break; //we didn't get any data } oldSize = currentBufferSize(); } DWORD bytesRead = qMin(currentBufferSize(), int(length)); { QWriteLocker locker(&m_lock); qMemCopy(buffer, m_buffer.data(), bytesRead); //truncate the buffer m_buffer = m_buffer.mid(bytesRead); } if (actual) { *actual = bytesRead; //initialization } return bytesRead == length ? S_OK : S_FALSE; } public: //for Phonon::StreamInterface QByteArray m_buffer; bool m_seekable; qint64 m_pos; qint64 m_size; QMutex m_mutexRead; const MediaGraph *m_mediaGraph; }; IODeviceReader::IODeviceReader(const Phonon::MediaSource &source, const MediaGraph *mg) : QBaseFilter(CLSID_NULL) { //create the output pin m_streamReader = new StreamReader(this, source, mg); } IODeviceReader::~IODeviceReader() { } STDMETHODIMP IODeviceReader::Stop() { HRESULT hr = QBaseFilter::Stop(); m_streamReader->enoughData(); //this asks to cancel any blocked call to needData return hr; } } } #endif //QT_NO_PHONON_ABSTRACTMEDIASTREAM QT_END_NAMESPACE