blob: 160d23fbcb1c09ef1dcca7f4da70106078a086db [file] [log] [blame]
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Scale volume with slew rate limiting
#ifndef CHROMECAST_MEDIA_BASE_SLEW_VOLUME_H_
#define CHROMECAST_MEDIA_BASE_SLEW_VOLUME_H_
#include <stdint.h>
#include "base/macros.h"
namespace chromecast {
namespace media {
class SlewVolume {
public:
SlewVolume();
explicit SlewVolume(int max_slew_time_ms);
~SlewVolume() = default;
void SetSampleRate(int sample_rate);
void SetVolume(double volume_scale);
// Return the largest multiplier that was applied in the last call to
// ProcessFMAC() or ProcessFMUL(). The largest multiplier is used because
// that determines the largest possible value in the buffer.
float LastBufferMaxMultiplier();
void SetMaxSlewTimeMs(int max_slew_time_ms);
// Called to indicate that the stream was interrupted; volume changes can be
// applied immediately.
void Interrupted();
// Smoothly calculates dest[i] += src[i] * |volume_scale|.
// |volume_scale| will always be consistent across a frame.
// |src| and |dest| are interleaved buffers with |channels| channels and at
// least |frames| frames (|channels| * |frames| total size).
// |src| and |dest| may be the same.
// |src| and |dest| must be 16-byte aligned.
// If using planar data, |repeat_transition| should be true for channels 2
// through n, which will cause the slewing process to be repeated.
void ProcessFMAC(bool repeat_transition,
const float* src,
int frames,
int channels,
float* dest);
// Smoothly calculates dest[i] = src[i] * |volume_scale|.
// |volume_scale| will always be consistent across a frame.
// |src| and |dest| are interleaved buffers with |channels| channels and at
// least |frames| frames (|channels| * |frames| total size).
// |src| and |dest| may be the same.
// |src| and |dest| must be 16-byte aligned.
// If using planar data, |repeat_transition| should be true for channels 2
// through n, which will cause the slewing process to be repeated.
void ProcessFMUL(bool repeat_transition,
const float* src,
int frames,
int channels,
float* dest);
private:
template <typename Traits>
void ProcessData(bool repeat_transition,
const float* src,
int frames,
int channels,
float* dest);
double sample_rate_;
double volume_scale_ = 1.0;
double current_volume_ = 1.0;
double last_starting_volume_ = 1.0;
double max_slew_time_ms_;
double max_slew_per_sample_;
bool interrupted_ = true;
DISALLOW_COPY_AND_ASSIGN(SlewVolume);
};
} // namespace media
} // namespace chromecast
#endif // CHROMECAST_MEDIA_BASE_SLEW_VOLUME_H_