Fred Gleason 37da80e1c3 2019-08-06 Fred Gleason <fredg@paravelsystems.com>
* Refactored ripcd(8) to manage JACK port [dis]connections directly,
	rather than by delegation to caed(8).
2019-08-06 19:54:38 -04:00

437 lines
16 KiB
C++

// cae.h
//
// The Core Audio Engine component of Rivendell
//
// (C) Copyright 2002-2019 Fred Gleason <fredg@paravelsystems.com>
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License version 2 as
// published by the Free Software Foundation.
//
// This program 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 General Public License for more details.
//
// You should have received a copy of the GNU General Public
// License along with this program; if not, write to the Free Software
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
//
#ifndef CAE_H
#define CAE_H
#include <sys/types.h>
#include <pthread.h>
#include <stdint.h>
#include <soundtouch/SoundTouch.h>
#include <qobject.h>
#include <qstring.h>
#include <qsignalmapper.h>
#include <qtimer.h>
//#include <q3process.h>
#include <qprocess.h>
#include <qudpsocket.h>
#include <rdwavefile.h>
#ifdef HPI
#include <rdhpisoundcard.h>
#include <rdhpiplaystream.h>
#include <rdhpirecordstream.h>
#endif // HPI
#ifdef ALSA
#include <alsa/asoundlib.h>
struct alsa_format {
int card;
pthread_t thread;
snd_pcm_t *pcm;
unsigned channels;
unsigned capture_channels;
snd_pcm_uframes_t buffer_size;
snd_pcm_format_t format;
unsigned sample_rate;
char *card_buffer;
char *passthrough_buffer;
unsigned card_buffer_size;
unsigned periods;
bool exiting;
};
#endif // ALSA
#ifdef JACK
#include <jack/jack.h>
#endif // JACK
#ifdef HAVE_TWOLAME
#include <twolame.h>
#endif // HAVE_TWOLAME
#ifdef HAVE_MAD
#include <mad.h>
#endif // HAVE_MAD
#include <rd.h>
#include <rdconfig.h>
#include <rdstation.h>
#include "cae_server.h"
#ifndef HAVE_SRC_CONV
void src_int_to_float_array (const int *in, float *out, int len);
void src_float_to_int_array (const float *in, int *out, int len);
#endif // HAVE_SRC_CONV
//
// Debug Options
//
//#define PRINT_COMMANDS
//
// Global CAE Definitions
//
#define RINGBUFFER_SIZE 262144
#define CAED_USAGE "[-d]\n\nSupplying the '-d' flag will set 'debug' mode, causing caed(8) to stay\nin the foreground and print debugging info on standard output.\n"
//
// Function Prototypes
//
void SigHandler(int signum);
extern RDConfig *rd_config;
class MainObject : public QObject
{
Q_OBJECT
public:
MainObject(QObject *parent=0,const char *name=0);
private slots:
void loadPlaybackData(int id,unsigned card,const QString &name);
void unloadPlaybackData(int id,unsigned handle);
void playPositionData(int id,unsigned handle,unsigned pos);
void playData(int id,unsigned handle,unsigned length,unsigned speed,
unsigned pitch_flag);
void stopPlaybackData(int id,unsigned handle);
void timescalingSupportData(int id,unsigned card);
void loadRecordingData(int id,unsigned card,unsigned port,unsigned coding,
unsigned channels,unsigned samprate,unsigned bitrate,
const QString &name);
void unloadRecordingData(int id,unsigned card,unsigned stream);
void recordData(int id,unsigned card,unsigned stream,unsigned len,
int threshold_level);
void stopRecordingData(int id,unsigned card,unsigned stream);
void setInputVolumeData(int id,unsigned card,unsigned stream,int level);
void setOutputVolumeData(int id,unsigned card,unsigned stream,unsigned port,
int level);
void fadeOutputVolumeData(int id,unsigned card,unsigned stream,unsigned port,
int level,unsigned length);
void setInputLevelData(int id,unsigned card,unsigned stream,int level);
void setOutputLevelData(int id,unsigned card,unsigned port,int level);
void setInputModeData(int id,unsigned card,unsigned stream,unsigned mode);
void setOutputModeData(int id,unsigned card,unsigned stream,unsigned mode);
void setInputVoxLevelData(int id,unsigned card,unsigned stream,int level);
void setInputTypeData(int id,unsigned card,unsigned port,unsigned type);
void getInputStatusData(int id,unsigned card,unsigned port);
void setAudioPassthroughLevelData(int id,unsigned card,unsigned input,
unsigned output,int level);
void setClockSourceData(int id,unsigned card,int input);
void setOutputStatusFlagData(int id,unsigned card,unsigned port,
unsigned stream,bool state);
void openRtpCaptureChannelData(int id,unsigned card,unsigned port,
uint16_t udp_port,unsigned samprate,
unsigned chans);
void meterEnableData(int id,uint16_t udp_port,const QList<unsigned> &cards);
void statePlayUpdate(int card,int stream,int state);
void stateRecordUpdate(int card,int stream,int state);
void updateMeters();
void connectionDroppedData(int id);
private:
void InitProvisioning() const;
void InitMixers();
void KillSocket(int);
bool CheckDaemon(QString);
pid_t GetPid(QString pidfile);
int GetNextHandle();
int GetHandle(int card,int stream);
void ProbeCaps(RDStation *station);
void ClearDriverEntries(RDStation *station);
void SendMeterLevelUpdate(const QString &type,int cardnum,int portnum,
short levels[]);
void SendStreamMeterLevelUpdate(int cardnum,int streamnum,short levels[]);
void SendMeterPositionUpdate(int cardnum,unsigned pos[]);
void SendMeterOutputStatusUpdate();
void SendMeterOutputStatusUpdate(int card,int port,int stream);
void SendMeterUpdate(const QString &msg,int conn_id);
bool debug;
unsigned system_sample_rate;
CaeServer *cae_server;
Q_INT16 tcp_port;
QUdpSocket *meter_socket;
RDStation::AudioDriver cae_driver[RD_MAX_CARDS];
int record_owner[RD_MAX_CARDS][RD_MAX_STREAMS];
int record_length[RD_MAX_CARDS][RD_MAX_STREAMS];
int record_threshold[RD_MAX_CARDS][RD_MAX_STREAMS];
int play_owner[RD_MAX_CARDS][RD_MAX_STREAMS];
int play_length[RD_MAX_CARDS][RD_MAX_STREAMS];
int play_speed[RD_MAX_CARDS][RD_MAX_STREAMS];
bool play_pitch[RD_MAX_CARDS][RD_MAX_STREAMS];
bool port_status[RD_MAX_CARDS][RD_MAX_PORTS];
bool output_status_flag[RD_MAX_CARDS][RD_MAX_PORTS][RD_MAX_STREAMS];
struct {
int card;
int stream;
int owner;
} play_handle[256];
int next_play_handle;
RDStation *cae_station;
//
// HPI Driver
//
private:
void hpiInit(RDStation *station);
void hpiFree();
QString hpiVersion();
bool hpiLoadPlayback(int card,QString wavename,int *stream);
bool hpiUnloadPlayback(int card,int stream);
bool hpiPlaybackPosition(int card,int stream,unsigned pos);
bool hpiPlay(int card,int stream,int length,int speed,bool pitch,
bool rates);
bool hpiStopPlayback(int card,int stream);
bool hpiTimescaleSupported(int card);
bool hpiLoadRecord(int card,int port,int coding,int chans,int samprate,
int bitrate,QString wavename);
bool hpiUnloadRecord(int card,int stream,unsigned *len);
bool hpiRecord(int card,int stream,int length,int thres);
bool hpiStopRecord(int card,int stream);
bool hpiSetClockSource(int card,int src);
bool hpiSetInputVolume(int card,int stream,int level);
bool hpiSetOutputVolume(int card,int stream,int port,int level);
bool hpiFadeOutputVolume(int card,int stream,int port,int level,int length);
bool hpiSetInputLevel(int card,int port,int level);
bool hpiSetOutputLevel(int card,int port,int level);
bool hpiSetInputMode(int card,int stream,int mode);
bool hpiSetOutputMode(int card,int stream,int mode);
bool hpiSetInputVoxLevel(int card,int stream,int level);
bool hpiSetInputType(int card,int port,int type);
bool hpiGetInputStatus(int card,int port);
bool hpiGetInputMeters(int card,int port,short levels[2]);
bool hpiGetOutputMeters(int card,int port,short levels[2]);
bool hpiGetStreamOutputMeters(int card,int stream,short levels[2]);
bool hpiSetPassthroughLevel(int card,int in_port,int out_port,int level);
void hpiGetOutputPosition(int card,unsigned *pos);
#ifdef HPI
RDHPISoundCard *sound_card;
RDHPIRecordStream *record[RD_MAX_CARDS][RD_MAX_STREAMS];
RDHPIPlayStream *play[RD_MAX_CARDS][RD_MAX_STREAMS];
#endif // HPI
//
// JACK Driver
//
private slots:
void jackStopTimerData(int stream);
void jackFadeTimerData(int stream);
void jackRecordTimerData(int stream);
void jackClientStartData();
private:
void jackInit(RDStation *station);
void jackFree();
bool jackLoadPlayback(int card,QString wavename,int *stream);
bool jackUnloadPlayback(int card,int stream);
bool jackPlaybackPosition(int card,int stream,unsigned pos);
bool jackPlay(int card,int stream,int length,int speed,bool pitch,
bool rates);
bool jackStopPlayback(int card,int stream);
bool jackTimescaleSupported(int card);
bool jackLoadRecord(int card,int port,int coding,int chans,int samprate,
int bitrate,QString wavename);
bool jackUnloadRecord(int card,int stream,unsigned *len);
bool jackRecord(int card,int stream,int length,int thres);
bool jackStopRecord(int card,int stream);
bool jackSetInputVolume(int card,int stream,int level);
bool jackSetOutputVolume(int card,int stream,int port,int level);
bool jackFadeOutputVolume(int card,int stream,int port,int level,int length);
bool jackSetInputLevel(int card,int port,int level);
bool jackSetOutputLevel(int card,int port,int level);
bool jackSetInputMode(int card,int stream,int mode);
bool jackSetOutputMode(int card,int stream,int mode);
bool jackSetInputVoxLevel(int card,int stream,int level);
bool jackSetInputType(int card,int port,int type);
bool jackGetInputStatus(int card,int port);
bool jackGetInputMeters(int card,int port,short levels[2]);
bool jackGetOutputMeters(int card,int port,short levels[2]);
bool jackGetStreamOutputMeters(int card,int stream,short levels[2]);
bool jackSetPassthroughLevel(int card,int in_port,int out_port,int level);
void jackGetOutputPosition(int card,unsigned *pos);
void jackConnectPorts(const QString &out,const QString &in);
void jackDisconnectPorts(const QString &out,const QString &in);
int GetJackOutputStream();
void FreeJackOutputStream(int stream);
void EmptyJackInputStream(int stream,bool done);
#ifdef JACK
void WriteJackBuffer(int stream,jack_default_audio_sample_t *buffer,
unsigned len,bool done);
#endif // JACK
void FillJackOutputStream(int stream);
void JackClock();
void JackSessionSetup();
bool jack_connected;
bool jack_activated;
#ifdef JACK
int jack_card;
QList<QProcess *> jack_clients;
RDWaveFile *jack_record_wave[RD_MAX_STREAMS];
RDWaveFile *jack_play_wave[RD_MAX_STREAMS];
short *jack_wave_buffer;
int *jack_wave32_buffer;
uint8_t *jack_wave24_buffer;
jack_default_audio_sample_t *jack_sample_buffer;
soundtouch::SoundTouch *jack_st_conv[RD_MAX_STREAMS];
short jack_input_volume_db[RD_MAX_STREAMS];
short jack_output_volume_db[RD_MAX_PORTS][RD_MAX_STREAMS];
short jack_passthrough_volume_db[RD_MAX_PORTS][RD_MAX_PORTS];
short jack_fade_volume_db[RD_MAX_STREAMS];
short jack_fade_increment[RD_MAX_STREAMS];
int jack_fade_port[RD_MAX_STREAMS];
bool jack_fade_up[RD_MAX_STREAMS];
QTimer *jack_fade_timer[RD_MAX_STREAMS];
QTimer *jack_stop_timer[RD_MAX_STREAMS];
QTimer *jack_record_timer[RD_MAX_PORTS];
QTimer *jack_client_start_timer;
int jack_offset[RD_MAX_STREAMS];
int jack_clock_phase;
unsigned jack_samples_recorded[RD_MAX_STREAMS];
#endif // JACK
//
// ALSA Driver
//
private slots:
void alsaStopTimerData(int cardstream);
void alsaFadeTimerData(int cardstream);
void alsaRecordTimerData(int cardport);
private:
void alsaInit(RDStation *station);
void alsaFree();
bool alsaLoadPlayback(int card,QString wavename,int *stream);
bool alsaUnloadPlayback(int card,int stream);
bool alsaPlaybackPosition(int card,int stream,unsigned pos);
bool alsaPlay(int card,int stream,int length,int speed,bool pitch,
bool rates);
bool alsaStopPlayback(int card,int stream);
bool alsaTimescaleSupported(int card);
bool alsaLoadRecord(int card,int port,int coding,int chans,int samprate,
int bitrate,QString wavename);
bool alsaUnloadRecord(int card,int stream,unsigned *len);
bool alsaRecord(int card,int stream,int length,int thres);
bool alsaStopRecord(int card,int stream);
bool alsaSetInputVolume(int card,int stream,int level);
bool alsaSetOutputVolume(int card,int stream,int port,int level);
bool alsaFadeOutputVolume(int card,int stream,int port,int level,int length);
bool alsaSetInputLevel(int card,int port,int level);
bool alsaSetOutputLevel(int card,int port,int level);
bool alsaSetInputMode(int card,int stream,int mode);
bool alsaSetOutputMode(int card,int stream,int mode);
bool alsaSetInputVoxLevel(int card,int stream,int level);
bool alsaSetInputType(int card,int port,int type);
bool alsaGetInputStatus(int card,int port);
bool alsaGetInputMeters(int card,int port,short levels[2]);
bool alsaGetOutputMeters(int card,int port,short levels[2]);
bool alsaGetStreamOutputMeters(int card,int stream,short levels[2]);
bool alsaSetPassthroughLevel(int card,int in_port,int out_port,int level);
void alsaGetOutputPosition(int card,unsigned *pos);
void AlsaClock();
#ifdef ALSA
bool AlsaStartCaptureDevice(QString &dev,int card,snd_pcm_t *pcm);
bool AlsaStartPlayDevice(QString &dev,int card,snd_pcm_t *pcm);
void AlsaInitCallback();
int GetAlsaOutputStream(int card);
void FreeAlsaOutputStream(int card,int stream);
void EmptyAlsaInputStream(int card,int stream);
void WriteAlsaBuffer(int card,int stream,short *buffer,unsigned len);
void FillAlsaOutputStream(int card,int stream);
struct alsa_format alsa_play_format[RD_MAX_CARDS];
struct alsa_format alsa_capture_format[RD_MAX_CARDS];
short alsa_input_volume_db[RD_MAX_CARDS][RD_MAX_STREAMS];
short alsa_output_volume_db[RD_MAX_CARDS][RD_MAX_PORTS][RD_MAX_STREAMS];
short alsa_passthrough_volume_db[RD_MAX_CARDS][RD_MAX_PORTS][RD_MAX_PORTS];
short *alsa_wave_buffer;
uint8_t *alsa_wave24_buffer;
RDWaveFile *alsa_record_wave[RD_MAX_CARDS][RD_MAX_STREAMS];
RDWaveFile *alsa_play_wave[RD_MAX_CARDS][RD_MAX_STREAMS];
int alsa_offset[RD_MAX_CARDS][RD_MAX_STREAMS];
QTimer *alsa_fade_timer[RD_MAX_CARDS][RD_MAX_STREAMS];
QTimer *alsa_stop_timer[RD_MAX_CARDS][RD_MAX_STREAMS];
QTimer *alsa_record_timer[RD_MAX_CARDS][RD_MAX_PORTS];
bool alsa_fade_up[RD_MAX_CARDS][RD_MAX_STREAMS];
short alsa_fade_volume_db[RD_MAX_CARDS][RD_MAX_STREAMS];
short alsa_fade_increment[RD_MAX_CARDS][RD_MAX_STREAMS];
int alsa_fade_port[RD_MAX_CARDS][RD_MAX_STREAMS];
unsigned alsa_samples_recorded[RD_MAX_CARDS][RD_MAX_STREAMS];
#endif // ALSA
bool CheckLame();
bool CheckMp4Decode();
//
// TwoLAME Encoder
//
bool LoadTwoLame();
bool InitTwoLameEncoder(int card,int stream,int chans,int samprate,
int bitrate);
void FreeTwoLameEncoder(int card,int stream);
void *twolame_handle;
#ifdef HAVE_TWOLAME
twolame_options *(*twolame_init)(void);
void (*twolame_set_mode)(twolame_options *,TWOLAME_MPEG_mode);
void (*twolame_set_num_channels)(twolame_options *,int);
void (*twolame_set_in_samplerate)(twolame_options *,int);
void (*twolame_set_out_samplerate)(twolame_options *,int);
void (*twolame_set_bitrate)(twolame_options *,int);
int (*twolame_init_params)(twolame_options *);
void (*twolame_close)(twolame_options **);
int (*twolame_encode_buffer_interleaved)(twolame_options *,const short int[],
int,unsigned char *,int);
int (*twolame_encode_buffer_float32_interleaved)
(twolame_options *,const float[],int,unsigned char *,int);
int (*twolame_encode_flush)(twolame_options *,unsigned char *,int);
int (*twolame_set_energy_levels)(twolame_options *,int);
twolame_options *twolame_lameopts[RD_MAX_CARDS][RD_MAX_STREAMS];
#endif // HAVE_TWOLAME
//
// MAD Decoder
//
bool LoadMad();
void InitMadDecoder(int card,int stream,RDWaveFile *wave);
void FreeMadDecoder(int card,int stream);
void *mad_handle;
#ifdef HAVE_MAD
void (*mad_stream_init)(struct mad_stream *);
void (*mad_frame_init)(struct mad_frame *);
void (*mad_synth_init)(struct mad_synth *);
void (*mad_stream_buffer)(struct mad_stream *,unsigned char const *,
unsigned long);
int (*mad_frame_decode)(struct mad_frame *, struct mad_stream *);
void (*mad_synth_frame)(struct mad_synth *, struct mad_frame const *);
void (*mad_frame_finish)(struct mad_frame *);
void (*mad_stream_finish)(struct mad_stream *);
struct mad_stream mad_stream[RD_MAX_CARDS][RD_MAX_STREAMS];
struct mad_frame mad_frame[RD_MAX_CARDS][RD_MAX_STREAMS];
struct mad_synth mad_synth[RD_MAX_CARDS][RD_MAX_STREAMS];
bool mad_active[RD_MAX_CARDS][RD_MAX_STREAMS];
int mad_frame_size[RD_MAX_CARDS][RD_MAX_STREAMS];
int mad_left_over[RD_MAX_CARDS][RD_MAX_STREAMS];
unsigned char *mad_mpeg[RD_MAX_CARDS][RD_MAX_STREAMS];
#endif // HAVE_MAD
};
#endif // CAE_H