Implemented an AudioReader subclass "AudioVolumeConverter" which applies

a volume to the audio data. It ramps between a previous and the current volume
if necessary to smooth out the changes. The volume slider functionality is
thereby restored.


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@26066 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Stephan Aßmus 2008-06-21 18:15:48 +00:00
parent bdc47b1f97
commit 50450a0053
6 changed files with 250 additions and 18 deletions

View File

@ -159,8 +159,10 @@ Controller::SetTo(const entry_ref &ref)
BAutolock _(this);
if (fRef == ref) {
if (InitCheck() == B_OK) {
SetCurrentFrame(0);
StartPlaying();
}
return B_OK;
}
@ -233,8 +235,8 @@ Controller::SetTo(const entry_ref &ref)
printf("Controller::SetTo: %d audio track, %d video track\n",
AudioTrackCount(), VideoTrackCount());
mediaFileDeleter.Detach();
fMediaFile = mf;
mediaFileDeleter.Detach();
SelectAudioTrack(0);
SelectVideoTrack(0);
@ -269,13 +271,12 @@ Controller::SetTo(const entry_ref &ref)
preferredVideoFormat);
}
SetCurrentFrame(0);
_NotifyFileChanged();
SetCurrentFrame(0);
if (fAutoplay)
StartPlaying(true);
_NotifyFileChanged();
return B_OK;
}
@ -477,9 +478,7 @@ Controller::SetVolume(float value)
ToggleMute();
fVolume = value;
// TODO: apply to AutioProducer node
// if (fSoundOutput)
// fSoundOutput->SetVolume(fVolume);
fAudioSupplier->SetVolume(fVolume);
_NotifyVolumeChanged(fVolume);
}
@ -509,13 +508,10 @@ Controller::ToggleMute()
fMuted = !fMuted;
// TODO: apply to AudioProducer node
// if (fSoundOutput) {
// if (fMuted)
// fSoundOutput->SetVolume(0.0);
// else
// fSoundOutput->SetVolume(fVolume);
// }
if (fMuted)
fAudioSupplier->SetVolume(0.0);
else
fAudioSupplier->SetVolume(fVolume);
_NotifyMutedChanged(fMuted);

View File

@ -44,6 +44,7 @@ Application MediaPlayer :
AudioReader.cpp
AudioResampler.cpp
AudioSupplier.cpp
AudioVolumeConverter.cpp
# media_node_framework/video
VideoConsumer.cpp

View File

@ -0,0 +1,169 @@
/*
* Copyright © 2008 Stephan Aßmus <superstippi@gmx.de>
* All rights reserved. Distributed under the terms of the MIT licensce.
*/
#include "AudioVolumeConverter.h"
#include <stdio.h>
#include <string.h>
#include <MediaDefs.h>
//#define TRACE_AUDIO_CONVERTER
#ifdef TRACE_AUDIO_CONVERTER
# define TRACE(x...) printf(x)
#else
# define TRACE(x...)
#endif
AudioVolumeConverter::AudioVolumeConverter(AudioReader* source, float volume)
: AudioReader(),
fSource(NULL),
fVolume(volume),
fPreviousVolume(volume)
{
if (source && source->Format().type == B_MEDIA_RAW_AUDIO)
fFormat = source->Format();
else
source = NULL;
fSource = source;
}
AudioVolumeConverter::~AudioVolumeConverter()
{
}
template<typename SampleType>
static void
convert(SampleType* buffer, const int32 samples, const float volume,
const float rounding)
{
for (int32 i = 0; i < samples; i++) {
*buffer = (SampleType)(*buffer * volume + rounding);
buffer++;
}
}
template<typename SampleType>
static void
convert(SampleType* buffer, const int32 frames, const int32 channels,
const float volume1, const float volume2, const float rounding)
{
float volumeDiff = volume2 - volume1;
for (int32 i = 0; i < frames; i++) {
float volume = volume1 + volumeDiff * (i / (frames - 1));
for (int32 k = 0; k < channels; k++) {
*buffer = (SampleType)(*buffer * volume + rounding);
buffer++;
}
}
}
status_t
AudioVolumeConverter::Read(void* buffer, int64 pos, int64 frames)
{
TRACE("AudioVolumeConverter::Read(%p, %lld, %lld)\n", buffer, pos, frames);
status_t error = InitCheck();
if (error != B_OK) {
TRACE("AudioVolumeConverter::Read() done 1\n");
return error;
}
pos += fOutOffset;
status_t ret = fSource->Read(buffer, pos, frames);
if (fPreviousVolume == 1.0 && fVolume == 1.0) {
TRACE("AudioVolumeConverter::Read() done 2\n");
return ret;
}
int32 channelCount = fFormat.u.raw_audio.channel_count;
int32 samples = frames * channelCount;
// apply volume
switch (fSource->Format().u.raw_audio.format) {
case media_raw_audio_format::B_AUDIO_FLOAT:
if (fVolume != fPreviousVolume) {
convert((float*)buffer, frames, channelCount,
fPreviousVolume, fVolume, 0.0);
} else
convert((float*)buffer, samples, fVolume, 0.0);
break;
case media_raw_audio_format::B_AUDIO_INT:
if (fVolume != fPreviousVolume) {
convert((int32*)buffer, frames, channelCount,
fPreviousVolume, fVolume, 0.5);
} else
convert((int32*)buffer, samples, fVolume, 0.5);
break;
case media_raw_audio_format::B_AUDIO_SHORT:
if (fVolume != fPreviousVolume) {
convert((int16*)buffer, frames, channelCount,
fPreviousVolume, fVolume, 0.5);
} else
convert((int16*)buffer, samples, fVolume, 0.5);
break;
case media_raw_audio_format::B_AUDIO_UCHAR: {
// handle this extra, because center != 0
// (also ignores ramping the volume)
uchar* b = (uchar*)buffer;
for (int32 i = 0; i < samples; i++) {
*b = (uchar)(((float)*b - 128) * fVolume + 128.5);
b++;
}
break;
}
case media_raw_audio_format::B_AUDIO_CHAR:
if (fVolume != fPreviousVolume) {
convert((int8*)buffer, frames, channelCount,
fPreviousVolume, fVolume, 0.0);
} else
convert((int8*)buffer, samples, fVolume, 0.5);
break;
}
fPreviousVolume = fVolume;
TRACE("AudioVolumeConverter::Read() done\n");
return B_OK;
}
status_t
AudioVolumeConverter::InitCheck() const
{
status_t error = AudioReader::InitCheck();
if (error == B_OK && !fSource)
error = B_NO_INIT;
if (error == B_OK)
error = fSource->InitCheck();
return error;
}
AudioReader*
AudioVolumeConverter::Source() const
{
return fSource;
}
void
AudioVolumeConverter::SetVolume(float volume)
{
fVolume = volume;
}
float
AudioVolumeConverter::Volume() const
{
return fVolume;
}

View File

@ -0,0 +1,36 @@
/*
* Copyright © 2008 Stephan Aßmus <superstippi@gmx.de>
* All rights reserved. Distributed under the terms of the MIT licensce.
*/
/*! This AudioReader just filters the volume. It depends on floating point
* audio format.
*/
#ifndef AUDIO_VOLUME_CONVERTER_H
#define AUDIO_VOLUME_CONVERTER_H
#include "AudioReader.h"
class AudioVolumeConverter : public AudioReader {
public:
AudioVolumeConverter(AudioReader* source,
float volume = 1.0);
virtual ~AudioVolumeConverter();
virtual status_t Read(void* buffer, int64 pos, int64 frames);
virtual status_t InitCheck() const;
AudioReader* Source() const;
void SetVolume(float volume);
float Volume() const;
protected:
AudioReader* fSource;
float fVolume;
float fPreviousVolume;
};
#endif // AUDIO_VOLUME_CONVERTER_H

View File

@ -14,6 +14,7 @@
#include "AudioTrackSupplier.h"
#include "AudioAdapter.h"
#include "AudioVolumeConverter.h"
#include "PlaybackManager.h"
using std::nothrow;
@ -49,9 +50,11 @@ ProxyAudioSupplier::ProxyAudioSupplier(PlaybackManager* playbackManager)
, fPlaybackManager(playbackManager)
, fVideoFrameRate(25.0)
, fVolume(1.0)
, fSupplier(NULL)
, fAdapter(NULL)
, fVolumeConverter(NULL)
, fAudioResampler()
{
TRACE("ProxyAudioSupplier()\n");
@ -62,6 +65,7 @@ ProxyAudioSupplier::~ProxyAudioSupplier()
{
TRACE("~ProxyAudioSupplier()\n");
delete fAdapter;
delete fVolumeConverter;
}
@ -224,9 +228,30 @@ ProxyAudioSupplier::SetSupplier(AudioTrackSupplier* supplier,
fVideoFrameRate = videoFrameRate;
delete fAdapter;
fAdapter = new AudioAdapter(fSupplier, Format());
delete fVolumeConverter;
fAudioResampler.SetSource(fAdapter);
fAdapter = new AudioAdapter(fSupplier, Format());
fVolumeConverter = new AudioVolumeConverter(fAdapter, fVolume);
fAudioResampler.SetSource(fVolumeConverter);
}
void
ProxyAudioSupplier::SetVolume(float volume)
{
BAutolock _(fSupplierLock);
fVolume = volume;
if (fVolumeConverter)
fVolumeConverter->SetVolume(volume);
}
float
ProxyAudioSupplier::Volume()
{
BAutolock _(fSupplierLock);
return fVolume;
}

View File

@ -12,6 +12,7 @@
class AudioTrackSupplier;
class AudioVolumeConverter;
class PlaybackManager;
@ -33,6 +34,8 @@ public:
// ProxyAudioSupplier
void SetSupplier(AudioTrackSupplier* supplier,
float videoFrameRate);
void SetVolume(float volume);
float Volume();
private:
int64 _AudioFrameForVideoFrame(int64 frame) const;
@ -50,9 +53,11 @@ private:
PlaybackManager* fPlaybackManager;
float fVideoFrameRate;
float fVolume;
AudioTrackSupplier* fSupplier;
AudioReader* fAdapter;
AudioVolumeConverter* fVolumeConverter;
AudioResampler fAudioResampler;
};