mirror of
https://review.haiku-os.org/haiku
synced 2025-01-26 00:04:54 +01:00
59b4be8cc1
Signed-off-by: Jérôme Duval <jerome.duval@gmail.com>
152 lines
5.1 KiB
C++
152 lines
5.1 KiB
C++
/*******************************************************************************
|
|
/
|
|
/ File: SoundConsumer.h
|
|
/
|
|
/ Description: Record sound from some sound-producing Node.
|
|
/
|
|
/ Copyright 1998, Be Incorporated, All Rights Reserved
|
|
/
|
|
*******************************************************************************/
|
|
#ifndef SOUND_CONSUMER_H
|
|
#define SOUND_CONSUMER_H
|
|
|
|
// To use this Consumer:
|
|
// 1. Create Record and Notify hooks, or subclass SoundConsumer
|
|
// if you'd rather use the inheritance hierarchy.
|
|
// * The Record function should do whatever you want to do
|
|
// when you receive a buffer.
|
|
// * The Notify function should handle whatever events
|
|
// you wish to handle (defined in SoundUtil.h).
|
|
// 2: Create an instance of SoundConsumer, giving it the
|
|
// appropriate hook functions. Or, create an instance of an
|
|
// appropriate subclass if you've made one.
|
|
// 3: Register your new Consumer with the MediaRoster.
|
|
// 4: Connect your Consumer to some Producer.
|
|
// 5: Start or Stop the Consumer if your hook functions
|
|
// implement behavior for these kinds of events.
|
|
// Seek the Consumer to set the offset of the timestamps that
|
|
// your Record function will see.
|
|
// 6: When you're done, disconnect the Consumer, then delete it.
|
|
|
|
#include <BufferConsumer.h>
|
|
#include "SoundUtils.h"
|
|
|
|
namespace BPrivate {
|
|
|
|
|
|
class SoundConsumer : public BBufferConsumer {
|
|
public:
|
|
SoundConsumer(const char* name,
|
|
SoundProcessFunc recordFunc = NULL,
|
|
SoundNotifyFunc notifyFunc = NULL,
|
|
void* cookie = NULL);
|
|
virtual ~SoundConsumer();
|
|
|
|
//This function is OK to call from any thread.
|
|
status_t SetHooks(SoundProcessFunc recordFunc = NULL,
|
|
SoundNotifyFunc notifyFunc = NULL,
|
|
void* cookie = NULL);
|
|
|
|
// The MediaNode interface
|
|
public:
|
|
virtual port_id ControlPort() const;
|
|
virtual BMediaAddOn* AddOn(int32* internalID) const;
|
|
// Who instantiated you -- or NULL for app class
|
|
|
|
|
|
protected:
|
|
virtual void Start(bigtime_t performanceTime);
|
|
virtual void Stop(bigtime_t performanceTime, bool immediate);
|
|
virtual void Seek(bigtime_t mediaTime,
|
|
bigtime_t performanceTime);
|
|
virtual void SetRunMode(run_mode mode);
|
|
virtual void TimeWarp(bigtime_t atRealTime,
|
|
bigtime_t to_performanceTime);
|
|
virtual void Preroll();
|
|
virtual void SetTimeSource(BTimeSource* timeSource);
|
|
virtual status_t HandleMessage(int32 message, const void* data,
|
|
size_t size);
|
|
|
|
// The BufferConsumer interface
|
|
virtual status_t AcceptFormat(const media_destination& dest,
|
|
media_format* format);
|
|
virtual status_t GetNextInput(int32* cookie,
|
|
media_input* _input);
|
|
virtual void DisposeInputCookie(int32 cookie);
|
|
virtual void BufferReceived(BBuffer* buffer);
|
|
virtual void ProducerDataStatus(
|
|
const media_destination& forWhom,
|
|
int32 status, bigtime_t atMediaTime);
|
|
virtual status_t GetLatencyFor(const media_destination& forWhom,
|
|
bigtime_t* _latency,
|
|
media_node_id* _timesource);
|
|
virtual status_t Connected(const media_source& producer,
|
|
const media_destination& where,
|
|
const media_format& format,
|
|
media_input* _input);
|
|
virtual void Disconnected(const media_source& producer,
|
|
const media_destination& where);
|
|
virtual status_t FormatChanged(const media_source& producer,
|
|
const media_destination& consumer,
|
|
int32 fromChangeCount,
|
|
const media_format& format);
|
|
|
|
protected:
|
|
// Functions called when no hooks are installed.
|
|
// OK to override instead of installing hooks.
|
|
virtual void Record(bigtime_t time, const void* data,
|
|
size_t size,
|
|
const media_raw_audio_format& format);
|
|
virtual void Notify(int32 cause, ...);
|
|
|
|
private:
|
|
SoundProcessFunc m_recordHook;
|
|
SoundNotifyFunc m_notifyHook;
|
|
void* m_cookie;
|
|
media_input m_input;
|
|
thread_id m_thread;
|
|
port_id m_port;
|
|
|
|
// The times we need to deal with
|
|
// My notation for times: tr = real time,
|
|
// tp = performance time, tm = media time.
|
|
bigtime_t m_trTimeout;
|
|
// how long to wait on the input port
|
|
bigtime_t m_tpSeekAt;
|
|
// when we Seek
|
|
bigtime_t m_tmSeekTo;
|
|
// target time for Seek
|
|
|
|
// The transformation from media to peformance time.
|
|
// d = p - m, so m + d = p.
|
|
// Media time is generally governed by the Seek
|
|
// function. In our node, we simply use media time as
|
|
// the time that we report to the record hook function.
|
|
// If we were a producer node, we might use media time
|
|
// to track where we were in playing a certain piece
|
|
// of media. But we aren't.
|
|
bigtime_t m_delta;
|
|
|
|
// State variables
|
|
bool m_seeking;
|
|
// a Seek is pending
|
|
|
|
// Functions to calculate timing values. OK to override.
|
|
// ProcessingLatency is the time it takes to process a buffer;
|
|
// TotalLatency is returned to producer; Timeout is passed
|
|
// to call to read_port_etc() in service thread loop.
|
|
virtual bigtime_t Timeout();
|
|
virtual bigtime_t ProcessingLatency();
|
|
virtual bigtime_t TotalLatency();
|
|
// The actual thread doing the work
|
|
static status_t ThreadEntry(void* cookie);
|
|
void ServiceThread();
|
|
void DoHookChange(void* message);
|
|
};
|
|
|
|
|
|
}
|
|
|
|
|
|
#endif // SOUND_CONSUMER_H
|