1180 lines
37 KiB
C++
1180 lines
37 KiB
C++
/* PdVst v0.0.2 - VST - Pd bridging plugin
|
|
** Copyright (C) 2004 Joseph A. Sarlo
|
|
**
|
|
** This program is free software; you can redistribute it and/orsig
|
|
** modify it under the terms of the GNU General Public License
|
|
** as published by the Free Software Foundation; either version 2
|
|
** of the License, or (at your option) any later version.
|
|
**
|
|
** 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
**
|
|
** jsarlo@ucsd.edu
|
|
*/
|
|
|
|
#define _WIN32_WINDOWS 0x410
|
|
|
|
#include "pdvst.hpp"
|
|
#include <malloc.h>
|
|
#include <process.h>
|
|
#include <windows.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <math.h>
|
|
#include <time.h>
|
|
#include <ctype.h>
|
|
#include <sys/stat.h>
|
|
#include <fstream>
|
|
#include <unistd.h>
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
#define stat _stat
|
|
#endif
|
|
|
|
extern HINSTANCE hInstance;
|
|
extern bool globalDebug;
|
|
extern int globalNChannels;
|
|
extern int globalNPrograms;
|
|
extern int globalNParams;
|
|
extern long globalPluginId;
|
|
extern int globalNExternalLibs;
|
|
extern char globalExternalLib[MAXEXTERNS][MAXSTRLEN];
|
|
extern char globalVstParamName[MAXPARAMETERS][MAXSTRLEN];
|
|
extern char globalPluginPath[MAXFILENAMELEN];
|
|
extern char globalVstPluginPath[MAXFILENAMELEN]; // chemin d'accès dans le dossier vst contenant la dll)
|
|
extern char globalPluginName[MAXSTRLEN];
|
|
extern char globalPdFile[MAXFILENAMELEN];
|
|
extern char globalPureDataPath[MAXFILENAMELEN];
|
|
extern char globalHostPdvstPath[MAXFILENAMELEN];
|
|
extern bool globalCustomGui;
|
|
extern int globalCustomGuiWidth;
|
|
extern int globalCustomGuiHeight;
|
|
extern bool globalProgramsAreChunks;
|
|
|
|
extern bool globalIsASynth;
|
|
extern pdvstProgram globalProgram[MAXPROGRAMS];
|
|
|
|
int pdvst::referenceCount = 0;
|
|
|
|
|
|
/* change '/' characters to the system's native file separator */
|
|
void sys_bashfilename(const char *from, char *to)
|
|
{
|
|
char c;
|
|
while ((c = *from++))
|
|
{
|
|
#ifdef _WIN32
|
|
if (c == '/') c = '\\';
|
|
#endif
|
|
*to++ = c;
|
|
}
|
|
*to = 0;
|
|
}
|
|
/* change the system's native file separator to '/' characters */
|
|
void sys_unbashfilename(const char *from, char *to)
|
|
{
|
|
char c;
|
|
while ((c = *from++))
|
|
{
|
|
#ifdef _WIN32
|
|
if (c == '\\') c = '/';
|
|
#endif
|
|
*to++ = c;
|
|
}
|
|
*to = 0;
|
|
}
|
|
|
|
/*
|
|
bool fexists(const char *filename)
|
|
{
|
|
std::ifstream ifile(filename);
|
|
return ifile;
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
//-------------------------------------------------------------------------------------------------------
|
|
AudioEffect* createEffectInstance (audioMasterCallback audioMaster)
|
|
{
|
|
|
|
return new pdvst (audioMaster);
|
|
}
|
|
|
|
pdvst::pdvst(audioMasterCallback audioMaster)
|
|
:AudioEffectX(audioMaster, globalNPrograms, globalNParams)
|
|
{
|
|
// set debug output
|
|
debugFile = fopen("pdvstdebug.txt", "wt");
|
|
// debugFile = NULL;
|
|
|
|
// copy global data
|
|
isASynth = globalIsASynth;
|
|
customGui = globalCustomGui;
|
|
customGuiHeight = globalCustomGuiHeight;
|
|
customGuiWidth = globalCustomGuiWidth;
|
|
nChannels = globalNChannels;
|
|
nPrograms = globalNPrograms;
|
|
nParameters = globalNParams;
|
|
pluginId = globalPluginId;
|
|
nExternalLibs = globalNExternalLibs;
|
|
debugLog("name: %s", globalPluginName);
|
|
debugLog("synth: %d", globalIsASynth);
|
|
|
|
// VST setup
|
|
setNumInputs(nChannels);
|
|
setNumOutputs(nChannels);
|
|
setUniqueID(pluginId);
|
|
setInitialDelay(PDBLKSIZE * 2);
|
|
canProcessReplacing();
|
|
if (isASynth)
|
|
{
|
|
isSynth();
|
|
}
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
// initialize memory
|
|
vstParamName = new char*[MAXPARAMETERS];
|
|
for (i = 0; i < MAXPARAMETERS; i++)
|
|
vstParamName[i] = new char[MAXSTRLEN];
|
|
memset(vstParam, 0, MAXPARAMETERS * sizeof(float));
|
|
program = new pdvstProgram[MAXPROGRAMS];
|
|
|
|
for (i = 0; i < nExternalLibs; i++)
|
|
{
|
|
strcpy(externalLib[i], globalExternalLib[i]);
|
|
}
|
|
debugLog("channels: %d", nChannels);
|
|
audioBuffer = new pdVstBuffer(nChannels);
|
|
for (i = 0; i < MAXPARAMETERS; i++)
|
|
{
|
|
strcpy(vstParamName[i], globalVstParamName[i]);
|
|
}
|
|
strcpy(pluginPath, globalPluginPath);
|
|
strcpy(vstPluginPath, globalVstPluginPath);
|
|
strcpy(pluginName, globalPluginName);
|
|
strcpy(pdFile, globalPdFile);
|
|
debugLog("path: %s", pluginPath);
|
|
debugLog("nParameters = %d", nParameters);
|
|
debugLog("nPrograms = %d", nPrograms);
|
|
for (i = 0; i < nPrograms; i++)
|
|
{
|
|
strcpy(program[i].name, globalProgram[i].name);
|
|
for (j = 0; j < nParameters; j++)
|
|
{
|
|
program[i].paramValue[j] = globalProgram[i].paramValue[j];
|
|
}
|
|
debugLog(" %s", program[i].name);
|
|
}
|
|
// create editor
|
|
if (customGui)
|
|
{
|
|
editor = new pdvstEditor(this);
|
|
|
|
debugLog(" -has custom GUI-");
|
|
}
|
|
guiNameUpdated=false;
|
|
|
|
#ifdef VSTMIDIOUTENABLE
|
|
// initializing structures for midiout support
|
|
|
|
nbFramesWithoutMidiOutEvent=0;
|
|
|
|
evnts = (struct VstEvents*)malloc(sizeof(struct VstEvents) +
|
|
MAXMIDIOUTQUEUESIZE*sizeof(VstEvent*));
|
|
//initialisation du tableau midiEvnts[MAXMIDIOUTQUEUESIZE];
|
|
for (int k=0; k<MAXMIDIOUTQUEUESIZE;k++)
|
|
{
|
|
// Constants to all midi messages we send (NoteOn and NoteOff)
|
|
midiEvnts[k].type = kVstMidiType;
|
|
midiEvnts[k].byteSize = sizeof(VstMidiEvent);
|
|
midiEvnts[k].deltaFrames = 0;
|
|
midiEvnts[k].flags |= kVstMidiEventIsRealtime;
|
|
midiEvnts[k].detune = 0;
|
|
midiEvnts[k].noteOffVelocity = (char)0;
|
|
midiEvnts[k].reserved1 = 0;
|
|
midiEvnts[k].reserved2 = 0;
|
|
midiEvnts[k].midiData[3] = 0;
|
|
midiEvnts[k].noteOffset = 0;
|
|
|
|
//cast
|
|
evnts->events[k] = (VstEvent*)&midiEvnts[k];
|
|
}
|
|
|
|
|
|
#endif //VSTMIDIOUTENABLE
|
|
|
|
|
|
|
|
//timeBeginPeriod(1);
|
|
debugLog("startingPd...");
|
|
// start pd.exe
|
|
startPd();
|
|
debugLog("done");
|
|
//setProgram(curProgram);
|
|
referenceCount++;
|
|
|
|
// {JYG see pdvst::setProgram below for explanation
|
|
timeFromStartup=GetTickCount();
|
|
// JYG }
|
|
programsAreChunks(globalProgramsAreChunks);
|
|
sendPlugName(globalPluginName);
|
|
}
|
|
|
|
pdvst::~pdvst()
|
|
{
|
|
int i;
|
|
|
|
referenceCount--;
|
|
pdvstData->active = 0;
|
|
CloseHandle(pdvstTransferMutex);
|
|
UnmapViewOfFile(pdvstTransferFileMap);
|
|
CloseHandle(pdvstTransferFileMap);
|
|
for (i = 0; i < MAXPARAMETERS; i++)
|
|
delete vstParamName[i];
|
|
delete vstParamName;
|
|
delete program;
|
|
delete audioBuffer;
|
|
if (debugFile)
|
|
{
|
|
fclose(debugFile);
|
|
}
|
|
}
|
|
|
|
void pdvst::debugLog(char *fmt, ...)
|
|
{
|
|
va_list ap;
|
|
|
|
if (debugFile)
|
|
{
|
|
va_start(ap, fmt);
|
|
vfprintf(debugFile, fmt, ap);
|
|
va_end(ap);
|
|
putc('\n', debugFile);
|
|
fflush(debugFile);
|
|
}
|
|
}
|
|
|
|
void pdvst::startPd()
|
|
{
|
|
|
|
int i;
|
|
STARTUPINFO si;
|
|
PROCESS_INFORMATION pi;
|
|
char commandLineArgs[MAXSTRLEN],
|
|
debugString[MAXSTRLEN],pathToCheck[MAXSTRLEN];
|
|
|
|
ZeroMemory(&si, sizeof(si));
|
|
si.cb = sizeof(si);
|
|
ZeroMemory(&pi, sizeof(pi));
|
|
sprintf(pdvstTransferMutexName, "mutex%d%x", GetCurrentProcessId(), this);
|
|
sprintf(pdvstTransferFileMapName, "filemap%d%x", GetCurrentProcessId(), this);
|
|
sprintf(vstProcEventName, "vstprocevent%d%x", GetCurrentProcessId(), this);
|
|
sprintf(pdProcEventName, "pdprocevent%d%x", GetCurrentProcessId(), this);
|
|
pdvstTransferMutex = CreateMutex(NULL, 0, pdvstTransferMutexName);
|
|
vstProcEvent = CreateEvent(NULL, TRUE, TRUE, vstProcEventName);
|
|
pdProcEvent = CreateEvent(NULL, TRUE, FALSE, pdProcEventName);
|
|
pdvstTransferFileMap = CreateFileMapping(INVALID_HANDLE_VALUE,
|
|
NULL,
|
|
PAGE_READWRITE,
|
|
0,
|
|
sizeof(pdvstTransferData),
|
|
pdvstTransferFileMapName);
|
|
pdvstData = (pdvstTransferData *)MapViewOfFile(pdvstTransferFileMap,
|
|
FILE_MAP_ALL_ACCESS,
|
|
0,
|
|
0,
|
|
sizeof(pdvstTransferData));
|
|
pdvstData->active = 1;
|
|
pdvstData->blockSize = PDBLKSIZE;
|
|
pdvstData->nChannels = nChannels;
|
|
pdvstData->sampleRate = (int)getSampleRate();
|
|
pdvstData->nParameters = nParameters;
|
|
pdvstData->guiState.updated = 0;
|
|
pdvstData->guiState.type = FLOAT_TYPE;
|
|
pdvstData->guiState.direction = PD_RECEIVE;
|
|
for (i = 0; i < MAXPARAMETERS; i++)
|
|
{
|
|
pdvstData->vstParameters[i].direction = PD_RECEIVE;
|
|
pdvstData->vstParameters[i].updated = 0;
|
|
}
|
|
if (globalDebug)
|
|
{
|
|
strcpy(debugString, "");
|
|
}
|
|
else
|
|
{
|
|
strcpy(debugString, " -nogui");
|
|
}
|
|
if (strcmp(globalPureDataPath, "") == 0) // unspecified puredata path. Defaults to globalHostPdvstPath
|
|
{
|
|
sprintf(commandLineArgs, "%s\\pd\\bin\\pd.exe", globalHostPdvstPath);
|
|
if( access( commandLineArgs, F_OK ) != -1 ) // check if file exists
|
|
sprintf(commandLineArgs, "\"%s\\pd\\bin\\pd.exe\"", globalHostPdvstPath);
|
|
else
|
|
{
|
|
sprintf(commandLineArgs, "%s..\\.pd\\bin\\pd.exe", globalPluginPath);
|
|
//MessageBox(NULL,commandLineArgs,"commandLineArgs...",MB_OK);
|
|
if( access( commandLineArgs, F_OK ) != -1 ) // check if file exists
|
|
sprintf(commandLineArgs, "\"%s..\\.pd\\bin\\pd.exe\"", globalPluginPath);
|
|
else
|
|
{
|
|
sprintf(errorMessage,"pd.exe program not found in \n1: %spd\\bin\\\n2: %s..\\.pd\\bin\\\nCheck your settings in \n%s%s.pdv",
|
|
globalHostPdvstPath,globalPluginPath,globalPluginPath,globalPluginName);
|
|
MessageBox(NULL,errorMessage,"ERROR",MB_OK);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
sprintf(commandLineArgs, "%s\\bin\\pd.exe", globalPureDataPath);
|
|
if ( access( commandLineArgs, F_OK ) != -1 ) // file exists
|
|
sprintf(commandLineArgs,"\"%s\\bin\\pd.exe\"", globalPureDataPath);
|
|
else
|
|
{
|
|
sprintf(errorMessage,"pd.exe program not found. Check your settings in \n%s%s.pdv \n setup file. (search for the line PDPATH)",
|
|
globalPluginPath,globalPluginName);
|
|
MessageBox(NULL,errorMessage,"ERROR",MB_OK);
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
// if (!fexists(commandLineArgs))
|
|
// sprintf(commandLineArgs, "\"%s\\pd\\bin\\pd.exe\"", pluginPath);
|
|
// check validity of pd.exe path
|
|
/*
|
|
struct stat statbuf;
|
|
sys_bashfilename(commandLineArgs, pathToCheck);
|
|
if (stat(pathToCheck, &statbuf) < 0)
|
|
// if failed, use defaults
|
|
sprintf(commandLineArgs, "\"%s\\pd\\bin\\pd.exe\"", pluginPath);*/
|
|
|
|
|
|
sprintf(commandLineArgs,
|
|
"%s%s",
|
|
commandLineArgs,
|
|
debugString);
|
|
sprintf(commandLineArgs,
|
|
"%s -schedlib vstschedlib",
|
|
commandLineArgs);
|
|
sprintf(commandLineArgs,
|
|
"%s -extraflags \"-vstproceventname %s -pdproceventname %s -vsthostid %d -mutexname %s -filemapname %s\"",
|
|
commandLineArgs,
|
|
vstProcEventName,
|
|
pdProcEventName,
|
|
GetCurrentProcessId(),
|
|
pdvstTransferMutexName,
|
|
pdvstTransferFileMapName);
|
|
sprintf(commandLineArgs,
|
|
"%s -outchannels %d -inchannels %d",
|
|
commandLineArgs,
|
|
nChannels,
|
|
nChannels);
|
|
sprintf(commandLineArgs,
|
|
"%s -r %d",
|
|
commandLineArgs,
|
|
(int)getSampleRate());
|
|
|
|
sprintf(commandLineArgs,
|
|
"%s -open \"%s%s\"",
|
|
commandLineArgs,
|
|
pluginPath,
|
|
pdFile);
|
|
sprintf(commandLineArgs,
|
|
"%s -path \"%s\"",
|
|
commandLineArgs,
|
|
pluginPath);
|
|
for (i = 0; i < nExternalLibs; i++)
|
|
{
|
|
sprintf(commandLineArgs,
|
|
"%s -lib %s",
|
|
commandLineArgs,
|
|
externalLib[i]);
|
|
}
|
|
debugLog("command line: %s", commandLineArgs);
|
|
suspend();
|
|
CreateProcess(NULL,
|
|
commandLineArgs,
|
|
NULL,
|
|
NULL,
|
|
0,
|
|
0,
|
|
NULL,
|
|
NULL,
|
|
&si,
|
|
&pi);
|
|
}
|
|
|
|
void pdvst::sendGuiAction(int action)
|
|
{
|
|
WaitForSingleObject(pdvstTransferMutex, 10);// if (WaitForSingleObject(pdvstTransferMutex, 100)!= WAIT_TIMEOUT)
|
|
{
|
|
pdvstData->guiState.direction = PD_RECEIVE;
|
|
pdvstData->guiState.type = FLOAT_TYPE;
|
|
pdvstData->guiState.value.floatData = (float)action;
|
|
pdvstData->guiState.updated = 1;
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
}
|
|
|
|
void pdvst::sendPlugName(char * pName ) // pour envoyer le nom du plugin à puredata
|
|
{
|
|
|
|
WaitForSingleObject(pdvstTransferMutex, 10);//if (WaitForSingleObject(pdvstTransferMutex, 100)!= WAIT_TIMEOUT)
|
|
{
|
|
pdvstData->plugName.direction = PD_RECEIVE;
|
|
pdvstData->plugName.type = STRING_TYPE;
|
|
strcpy(pdvstData->plugName.value.stringData,pName);
|
|
pdvstData->plugName.updated = 1;
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
}
|
|
|
|
void pdvst::setSyncToVst(int value)
|
|
{
|
|
|
|
if (WaitForSingleObject(pdvstTransferMutex, 10)==WAIT_OBJECT_0)
|
|
{
|
|
if (pdvstData->syncToVst != value)
|
|
{
|
|
pdvstData->syncToVst = value;
|
|
}
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
|
|
}
|
|
|
|
void pdvst::suspend()
|
|
{
|
|
int i;
|
|
|
|
setSyncToVst(0);
|
|
SetEvent(vstProcEvent);
|
|
for (i = 0; i < audioBuffer->nChannels; i++)
|
|
{
|
|
memset(audioBuffer->in[i], 0, audioBuffer->size * sizeof(float));
|
|
memset(audioBuffer->out[i], 0, audioBuffer->size * sizeof(float));
|
|
}
|
|
audioBuffer->inFrameCount = audioBuffer->outFrameCount = 0;
|
|
dspActive = false;
|
|
setInitialDelay(PDBLKSIZE * 2);
|
|
}
|
|
|
|
void pdvst::resume()
|
|
{
|
|
int i;
|
|
|
|
setSyncToVst(1);
|
|
SetEvent(vstProcEvent);
|
|
for (i = 0; i < audioBuffer->nChannels; i++)
|
|
{
|
|
memset(audioBuffer->in[i], 0, audioBuffer->size * sizeof(float));
|
|
memset(audioBuffer->out[i], 0, audioBuffer->size * sizeof(float));
|
|
}
|
|
audioBuffer->inFrameCount = audioBuffer->outFrameCount = 0;
|
|
dspActive = true;
|
|
setInitialDelay(PDBLKSIZE * 2);
|
|
if (isASynth)
|
|
{
|
|
//wantEvents(); deprecated since VST 2.4
|
|
}
|
|
}
|
|
|
|
void pdvst::setProgram(VstInt32 prgmNum)
|
|
{
|
|
debugLog("appel de setProgram %d", prgmNum);
|
|
int i;
|
|
if (prgmNum >= 0 && prgmNum < nPrograms)
|
|
{
|
|
curProgram = prgmNum;
|
|
|
|
|
|
// {JYG to prevent host call of setProgram to override current param settings
|
|
// see https://www.kvraudio.com/forum/viewtopic.php?p=6391144
|
|
// sprintf(errorMessage,"GetTickCount=%d\ntimeFromStartup=%d",
|
|
// GetTickCount(),timeFromStartup);
|
|
//MessageBox(NULL, errorMessage,"setProgram",MB_OK);
|
|
if ((GetTickCount()-timeFromStartup) < 1000)
|
|
return;
|
|
else
|
|
{
|
|
for (i = 0; i < nParameters; i++)
|
|
{
|
|
setParameter(i, program[curProgram].paramValue[i]);
|
|
// MessageBox(NULL,"setvalue","setProgram",MB_OK);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
VstInt32 pdvst::getProgram()
|
|
{
|
|
return curProgram;
|
|
}
|
|
|
|
void pdvst::setProgramName(char *name)
|
|
{
|
|
strcpy(program[curProgram].name, name);
|
|
}
|
|
|
|
void pdvst::getProgramName(char *name)
|
|
{
|
|
strcpy(name, program[curProgram].name);
|
|
// {JYG see pdvst::setProgram below for explanation
|
|
timeFromStartup=GetTickCount();
|
|
// JYG }
|
|
}
|
|
|
|
void pdvst::setParameter(VstInt32 index, float value)
|
|
{
|
|
if (vstParam[index] != value)
|
|
{
|
|
vstParam[index] = value;
|
|
WaitForSingleObject(pdvstTransferMutex, 10);// if (WaitForSingleObject(pdvstTransferMutex, 100)!= WAIT_TIMEOUT)
|
|
{
|
|
pdvstData->vstParameters[index].type = FLOAT_TYPE;
|
|
pdvstData->vstParameters[index].value.floatData = value;
|
|
pdvstData->vstParameters[index].direction = PD_RECEIVE;
|
|
pdvstData->vstParameters[index].updated = 1;
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
}
|
|
}
|
|
|
|
float pdvst::getParameter(VstInt32 index)
|
|
{
|
|
if (index >= 0 && index < MAXPARAMETERS)
|
|
return vstParam[index];
|
|
else
|
|
return 0;
|
|
}
|
|
|
|
void pdvst::getParameterName(VstInt32 index, char *label)
|
|
{
|
|
if (index >= 0 && index < MAXPARAMETERS)
|
|
strcpy(label, vstParamName[index]);
|
|
}
|
|
|
|
void pdvst::getParameterDisplay(VstInt32 index, char *text)
|
|
{
|
|
if (index >= 0 && index < MAXPARAMETERS)
|
|
float2string(vstParam[index], text, 10);
|
|
}
|
|
|
|
void pdvst::getParameterLabel(VstInt32 index, char *label)
|
|
{
|
|
strcpy(label, "");
|
|
/* if (index >= 0 && index < MAXPARAMETERS)
|
|
strcpy(label, vstParamName[index]);*/
|
|
}
|
|
|
|
// FIXME
|
|
bool pdvst::getEffectName(char* name)
|
|
{
|
|
strcpy (name, globalPluginName);
|
|
return true;
|
|
}
|
|
|
|
// FIXME
|
|
bool pdvst::getOutputProperties(VstInt32 index, VstPinProperties* properties)
|
|
{
|
|
if (1)//index < 2)
|
|
{
|
|
sprintf (properties->label, "pd-vst %1d", index + 1);
|
|
properties->flags = kVstPinIsActive;
|
|
if ((index %2)==0)
|
|
properties->flags |= kVstPinIsStereo; // make channel 1+2 stereo
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
VstInt32 pdvst::getChunk (void** data, bool isPreset)
|
|
{
|
|
//MessageBox(NULL,"debug","getchunk",MB_OK);
|
|
WaitForSingleObject(pdvstTransferMutex, 10);
|
|
{
|
|
if(*data)
|
|
strcpy ((char *)*data, pdvstData->datachunk.value.stringData);
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
return strlen(pdvstData->datachunk.value.stringData);
|
|
}
|
|
|
|
VstInt32 pdvst::setChunk (void* data, VstInt32 byteSize, bool isPreset)
|
|
{
|
|
//MessageBox(NULL,"debug","setchunk",MB_OK);
|
|
WaitForSingleObject(pdvstTransferMutex, 10);
|
|
{
|
|
pdvstData->datachunk.direction = PD_RECEIVE;
|
|
pdvstData->datachunk.type = STRING_TYPE;
|
|
memset(&pdvstData->datachunk.value.stringData, '\0', MAXSTRINGSIZE);
|
|
strncpy(pdvstData->datachunk.value.stringData,(char *)data, (size_t)byteSize);
|
|
pdvstData->datachunk.updated = 1;
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
VstInt32 pdvst::canDo(char* text)
|
|
{
|
|
//if (isASynth)
|
|
if (1)
|
|
{
|
|
if (!strcmp (text, "sendVstEvents"))
|
|
return 1;
|
|
if (!strcmp (text, "sendVstMidiEvent"))
|
|
return 1;
|
|
if (!strcmp (text, "receiveVstEvents"))
|
|
return 1;
|
|
if (!strcmp (text, "receiveVstMidiEvent"))
|
|
return 1;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
void pdvst::process(float **input, float **output, VstInt32 sampleFrames)
|
|
{
|
|
#ifdef VSTMIDIOUTENABLE
|
|
|
|
WaitForSingleObject(pdvstTransferMutex, 10);//if (WaitForSingleObject(pdvstTransferMutex, 100)!= WAIT_TIMEOUT)
|
|
{
|
|
if (pdvstData->midiOutQueueUpdated)
|
|
{
|
|
// Faire une loop pour parcourir la midiqueue et recopier
|
|
//dans la structure midiOutEvents
|
|
int midiOutQueueSize=pdvstData->midiOutQueueSize;
|
|
for (int ka=0;ka<midiOutQueueSize;ka++)
|
|
{
|
|
midiEvnts[ka].midiData[0] = pdvstData->midiOutQueue[ka].statusByte;
|
|
midiEvnts[ka].midiData[1] = pdvstData->midiOutQueue[ka].dataByte1;
|
|
midiEvnts[ka].midiData[2] = pdvstData->midiOutQueue[ka].dataByte2;
|
|
}
|
|
evnts->numEvents=midiOutQueueSize;
|
|
|
|
sendVstEventsToHost(evnts); /// TODO : rajouter un test de réussite
|
|
pdvstData->midiOutQueueUpdated=0;
|
|
pdvstData->midiOutQueueSize=0;
|
|
|
|
evnts->numEvents=0;
|
|
}
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
|
|
#endif // VSTMIDIOUTENABLE
|
|
|
|
|
|
/* // Get play head information
|
|
VstTimeInfo * info = getTimeInfo(
|
|
kVstTempoValid |
|
|
kVstPpqPosValid |
|
|
kVstTransportPlaying |
|
|
kVstTransportChanged |
|
|
kVstTransportCycleActive |
|
|
kVstTransportRecording |
|
|
kVstAutomationWriting |
|
|
kVstAutomationReading |
|
|
0);
|
|
|
|
/* if(infos)
|
|
{
|
|
m_playing_list.setFloat(0, m_playinfos.isPlaying);
|
|
m_playing_list.setFloat(1, m_playinfos.timeInSeconds);
|
|
sendMessageAnything(m_patch_tie, s_playing, m_playing_list);
|
|
m_measure_list.setFloat(0, m_playinfos.bpm);
|
|
m_measure_list.setFloat(1, m_playinfos.timeSigNumerator);
|
|
m_measure_list.setFloat(2, m_playinfos.timeSigDenominator);
|
|
m_measure_list.setFloat(3, m_playinfos.ppqPosition);
|
|
m_measure_list.setFloat(4, m_playinfos.ppqPositionOfLastBarStart);
|
|
sendMessageAnything(m_patch_tie, s_measure, m_measure_list);
|
|
|
|
*/
|
|
|
|
int i, j, k, l;
|
|
int framesOut = 0;
|
|
|
|
if (!dspActive)
|
|
{
|
|
resume();
|
|
}
|
|
else
|
|
{
|
|
setSyncToVst(1);
|
|
}
|
|
|
|
|
|
for (i = 0; i < sampleFrames; i++)
|
|
{
|
|
for (j = 0; j < audioBuffer->nChannels; j++)
|
|
{
|
|
audioBuffer->in[j][audioBuffer->inFrameCount] = input[j][i];
|
|
}
|
|
(audioBuffer->inFrameCount)++;
|
|
// if enough samples to process then do it
|
|
if (audioBuffer->inFrameCount >= PDBLKSIZE)
|
|
{
|
|
audioBuffer->inFrameCount = 0;
|
|
updatePdvstParameters();
|
|
// wait for pd process event
|
|
// { JYG
|
|
//WaitForSingleObject(pdProcEvent, 10000);///int couldSync = WaitForSingleObject(pdProcEvent, 100)!= WAIT_TIMEOUT;
|
|
WaitForSingleObject(pdProcEvent,10);
|
|
//WaitForSingleObject(pdProcEvent, 10000);
|
|
/// JYG: 10000 était une valeur trop élevée qui faisait planter ableton live 8.04, va savoir pourquoi
|
|
// JYG }
|
|
///if (couldSync)
|
|
ResetEvent(pdProcEvent);
|
|
|
|
for (k = 0; k < PDBLKSIZE; k++)
|
|
{
|
|
for (l = 0; l < audioBuffer->nChannels; l++)
|
|
{
|
|
while (audioBuffer->outFrameCount >= audioBuffer->size)
|
|
{
|
|
audioBuffer->resize(audioBuffer->size * 2);
|
|
}
|
|
// get pd processed samples
|
|
///if (couldSync)
|
|
{
|
|
// get pd processed samples
|
|
audioBuffer->out[l][audioBuffer->outFrameCount] = pdvstData->samples[l][k];
|
|
// put new samples in for processing
|
|
pdvstData->samples[l][k] = audioBuffer->in[l][k];
|
|
}
|
|
|
|
}
|
|
(audioBuffer->outFrameCount)++;
|
|
}
|
|
///if (couldSync)
|
|
pdvstData->sampleRate = (int)getSampleRate();
|
|
// signal vst process event
|
|
SetEvent(vstProcEvent);
|
|
}
|
|
}
|
|
// output pd processed samples
|
|
for (i = 0; i < sampleFrames; i++)
|
|
{
|
|
for (j = 0; j < audioBuffer->nChannels; j++)
|
|
{
|
|
if (audioBuffer->outFrameCount > 0)
|
|
{
|
|
output[j][i] = audioBuffer->out[j][i];
|
|
}
|
|
else
|
|
{
|
|
output[j][i] = 0;
|
|
}
|
|
}
|
|
if (audioBuffer->outFrameCount > 0)
|
|
{
|
|
audioBuffer->outFrameCount--;
|
|
framesOut++;
|
|
}
|
|
}
|
|
// shift any remaining buffered out samples
|
|
if (audioBuffer->outFrameCount > 0)
|
|
{
|
|
for (i = 0; i < audioBuffer->nChannels; i++)
|
|
{
|
|
memmove(&(audioBuffer->out[i][0]),
|
|
&(audioBuffer->out[i][framesOut]),
|
|
(audioBuffer->outFrameCount) * sizeof(float));
|
|
}
|
|
}
|
|
}
|
|
|
|
DWORD WINAPI NotifySyncError(LPVOID lpParam) {
|
|
MessageBoxA(NULL, "Le processus PureData met du temps à répondre", "PdVST", MB_OK);
|
|
return 0;
|
|
}
|
|
|
|
void pdvst::processReplacing(float **input, float **output, VstInt32 sampleFrames)
|
|
{
|
|
#ifdef VSTMIDIOUTENABLE
|
|
|
|
if (WaitForSingleObject(pdvstTransferMutex, 10)==WAIT_OBJECT_0)
|
|
{
|
|
if (pdvstData->midiOutQueueUpdated)
|
|
{
|
|
// Faire une loop pour parcourir la midiqueue et recopier
|
|
//dans la structure midiOutEvents
|
|
int midiOutQueueSize=pdvstData->midiOutQueueSize;
|
|
for (int ka=0;ka<midiOutQueueSize;ka++)
|
|
{
|
|
midiEvnts[ka].midiData[0] = pdvstData->midiOutQueue[ka].statusByte;
|
|
midiEvnts[ka].midiData[1] = pdvstData->midiOutQueue[ka].dataByte1;
|
|
midiEvnts[ka].midiData[2] = pdvstData->midiOutQueue[ka].dataByte2;
|
|
}
|
|
evnts->numEvents=midiOutQueueSize;
|
|
|
|
sendVstEventsToHost(evnts); // rajouter un test de réussite
|
|
pdvstData->midiOutQueueUpdated=0;
|
|
pdvstData->midiOutQueueSize=0;
|
|
|
|
evnts->numEvents=0;
|
|
}
|
|
else
|
|
{
|
|
// Envoyer périodiquement un sendVstEventsToHost(no_event) pour signifier
|
|
// à Ableton Live qu'on est bien vivant et qu'on envoie du midi
|
|
// (tous les 120 tours de buffer)
|
|
nbFramesWithoutMidiOutEvent++;
|
|
if (nbFramesWithoutMidiOutEvent>120)
|
|
{
|
|
nbFramesWithoutMidiOutEvent=0;
|
|
midiEvnts[0].midiData[0]=0;
|
|
midiEvnts[0].midiData[1]=0;
|
|
midiEvnts[0].midiData[2]=0;
|
|
evnts->numEvents=1;
|
|
sendVstEventsToHost(evnts);
|
|
}
|
|
|
|
}
|
|
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
|
|
#endif // VSTMIDIOUTENABLE
|
|
|
|
|
|
// Get play head information
|
|
VstTimeInfo * infos = getTimeInfo(
|
|
kVstTempoValid |
|
|
kVstPpqPosValid |
|
|
kVstTransportPlaying |
|
|
kVstTransportChanged |
|
|
kVstTransportCycleActive |
|
|
kVstTransportRecording |
|
|
kVstAutomationWriting |
|
|
kVstAutomationReading |
|
|
0);
|
|
|
|
if(infos)
|
|
{
|
|
if (WaitForSingleObject(pdvstTransferMutex, 10)==WAIT_OBJECT_0)
|
|
{
|
|
|
|
pdvstData->hostTimeInfo.updated=1;
|
|
//copying vstTimeInfo into pdvst transfer data structure
|
|
pdvstData->hostTimeInfo.samplePos=infos->samplePos; ///< current Position in audio samples (always valid)
|
|
pdvstData->hostTimeInfo.sampleRate=infos->sampleRate; ///< current Sample Rate in Herz (always valid)
|
|
pdvstData->hostTimeInfo.nanoSeconds=infos->nanoSeconds; ///< System Time in nanoseconds (10^-9 second)
|
|
pdvstData->hostTimeInfo.ppqPos=infos->ppqPos; ///< Musical Position, in Quarter Note (1.0 equals 1 Quarter Note)
|
|
pdvstData->hostTimeInfo.tempo=infos->tempo; ///< current Tempo in BPM (Beats Per Minute)
|
|
pdvstData->hostTimeInfo.barStartPos=infos->barStartPos; ///< last Bar Start Position, in Quarter Note
|
|
pdvstData->hostTimeInfo.cycleStartPos=infos->cycleStartPos; ///< Cycle Start (left locator), in Quarter Note
|
|
pdvstData->hostTimeInfo.cycleEndPos=infos->cycleEndPos; ///< Cycle End (right locator), in Quarter Note
|
|
pdvstData->hostTimeInfo.timeSigNumerator=(int)infos->timeSigNumerator; ///< Time Signature Numerator (e.g. 3 for 3/4)
|
|
pdvstData->hostTimeInfo.timeSigDenominator=(int)infos->timeSigDenominator; ///< Time Signature Denominator (e.g. 4 for 3/4)
|
|
pdvstData->hostTimeInfo.smpteOffset=(int)infos->smpteOffset; ///< SMPTE offset (in SMPTE subframes (bits; 1/80 of a frame)). The current SMPTE position can be calculated using #samplePos, #sampleRate, and #smpteFrameRate.
|
|
pdvstData->hostTimeInfo.smpteFrameRate=(int)infos->smpteFrameRate; ///< @see VstSmpteFrameRate
|
|
pdvstData->hostTimeInfo.samplesToNextClock=(int)infos->samplesToNextClock; ///< MIDI Clock Resolution (24 Per Quarter Note), can be negative (nearest clock)
|
|
pdvstData->hostTimeInfo.flags=(int)infos->flags;
|
|
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
}
|
|
|
|
int i, j, k, l;
|
|
int framesOut = 0;
|
|
|
|
if (!dspActive)
|
|
{
|
|
resume();
|
|
}
|
|
else
|
|
{
|
|
setSyncToVst(1);
|
|
}
|
|
|
|
|
|
for (i = 0; i < sampleFrames; i++)
|
|
{
|
|
for (j = 0; j < audioBuffer->nChannels; j++)
|
|
{
|
|
audioBuffer->in[j][audioBuffer->inFrameCount] = input[j][i];
|
|
}
|
|
(audioBuffer->inFrameCount)++;
|
|
// if enough samples to process then do it
|
|
if (audioBuffer->inFrameCount >= PDBLKSIZE)
|
|
{
|
|
audioBuffer->inFrameCount = 0;
|
|
updatePdvstParameters();
|
|
// wait for pd process event
|
|
// { JYG
|
|
int gotPdProcEvent = (WaitForSingleObject(pdProcEvent, 10)==WAIT_OBJECT_0);
|
|
// WaitForSingleObject(pdProcEvent,100);
|
|
//WaitForSingleObject(pdProcEvent, 10000);
|
|
// JYG: 10000 était une valeur trop élevée qui faisait planter ableton live
|
|
// JYG }
|
|
if (gotPdProcEvent)
|
|
{
|
|
ResetEvent(pdProcEvent);
|
|
syncDefeatNumber=0;
|
|
}
|
|
else if (syncDefeatNumber==50)
|
|
{
|
|
//afficher messageErreur
|
|
//CreateThread(NULL, 0, &NotifySyncError, NULL, 0, NULL);
|
|
//startPd();
|
|
syncDefeatNumber++;
|
|
|
|
}
|
|
|
|
else
|
|
syncDefeatNumber++;
|
|
|
|
for (k = 0; k < PDBLKSIZE; k++)
|
|
{
|
|
for (l = 0; l < audioBuffer->nChannels; l++)
|
|
{
|
|
while (audioBuffer->outFrameCount >= audioBuffer->size)
|
|
{
|
|
audioBuffer->resize(audioBuffer->size * 2);
|
|
}
|
|
// get pd processed samples
|
|
if (gotPdProcEvent)
|
|
{
|
|
// get pd processed samples
|
|
audioBuffer->out[l][audioBuffer->outFrameCount] = pdvstData->samples[l][k];
|
|
// put new samples in for processing
|
|
pdvstData->samples[l][k] = audioBuffer->in[l][k];
|
|
}
|
|
|
|
}
|
|
(audioBuffer->outFrameCount)++;
|
|
}
|
|
pdvstData->sampleRate = (int)getSampleRate();
|
|
// signal vst process event
|
|
SetEvent(vstProcEvent);
|
|
}
|
|
}
|
|
// output pd processed samples
|
|
for (i = 0; i < sampleFrames; i++)
|
|
{
|
|
for (j = 0; j < audioBuffer->nChannels; j++)
|
|
{
|
|
if (audioBuffer->outFrameCount > 0)
|
|
{
|
|
output[j][i] = audioBuffer->out[j][i];
|
|
}
|
|
else
|
|
{
|
|
output[j][i] = 0;
|
|
}
|
|
}
|
|
if (audioBuffer->outFrameCount > 0)
|
|
{
|
|
audioBuffer->outFrameCount--;
|
|
framesOut++;
|
|
}
|
|
}
|
|
// shift any remaining buffered out samples
|
|
if (audioBuffer->outFrameCount > 0)
|
|
{
|
|
for (i = 0; i < audioBuffer->nChannels; i++)
|
|
{
|
|
memmove(&(audioBuffer->out[i][0]),
|
|
&(audioBuffer->out[i][framesOut]),
|
|
(audioBuffer->outFrameCount) * sizeof(float));
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
VstInt32 pdvst::processEvents(VstEvents* ev)
|
|
{
|
|
VstMidiEvent* event;
|
|
char* midiData;
|
|
long statusType;
|
|
long statusChannel;
|
|
|
|
WaitForSingleObject(pdvstTransferMutex, 10);/// if (WaitForSingleObject(pdvstTransferMutex, 100)!= WAIT_TIMEOUT)
|
|
// valeur d'attente courte (1000 au lieu de 10000 dans pdvst 0.2)
|
|
|
|
{
|
|
for (long i = 0; i < ev->numEvents; i++)
|
|
{
|
|
if ((ev->events[i])->type != kVstMidiType)
|
|
{
|
|
continue;
|
|
}
|
|
event = (VstMidiEvent*)ev->events[i];
|
|
midiData = event->midiData;
|
|
statusType = midiData[0] & 0xF0;
|
|
statusChannel = midiData[0] & 0x0F;
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].statusByte = midiData[0];
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].dataByte1 = midiData[1];
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].dataByte2 = midiData[2];
|
|
if (statusType == 0x80)
|
|
{
|
|
// note off
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].channelNumber = statusChannel;
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].messageType = NOTE_OFF;
|
|
}
|
|
else if (statusType == 0x90)
|
|
{
|
|
// note on
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].channelNumber = statusChannel;
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].messageType = NOTE_ON;
|
|
}
|
|
else if (statusType == 0xA0)
|
|
{
|
|
// key pressure
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].channelNumber = statusChannel;
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].messageType = KEY_PRESSURE;
|
|
}
|
|
else if (statusType == 0xB0)
|
|
{
|
|
// controller change
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].channelNumber = statusChannel;
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].messageType = CONTROLLER_CHANGE;
|
|
}
|
|
else if (statusType == 0xC0)
|
|
{
|
|
// program change
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].channelNumber = statusChannel;
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].messageType = PROGRAM_CHANGE;
|
|
}
|
|
else if (statusType == 0xD0)
|
|
{
|
|
// channel pressure
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].channelNumber = statusChannel;
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].messageType = CHANNEL_PRESSURE;
|
|
}
|
|
else if (statusType == 0xE0)
|
|
{
|
|
// pitch bend
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].channelNumber = statusChannel;
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].messageType = PITCH_BEND;
|
|
}
|
|
else
|
|
{
|
|
// something else
|
|
pdvstData->midiQueue[pdvstData->midiQueueSize].messageType = OTHER;
|
|
}
|
|
pdvstData->midiQueueSize++;
|
|
event++;
|
|
}
|
|
pdvstData->midiQueueUpdated = 1;
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
return 1;
|
|
}
|
|
///else
|
|
return 0;
|
|
}
|
|
|
|
void pdvst::updatePdvstParameters()
|
|
{
|
|
int i;
|
|
|
|
|
|
WaitForSingleObject(pdvstTransferMutex, 10);
|
|
// WaitForSingleObject(pdvstTransferMutex, 10000);///if (WaitForSingleObject(pdvstTransferMutex, 100)!= WAIT_TIMEOUT)
|
|
{
|
|
for (i = 0; i < pdvstData->nParameters; i++)
|
|
{
|
|
if (pdvstData->vstParameters[i].direction == PD_SEND && \
|
|
pdvstData->vstParameters[i].updated)
|
|
{
|
|
if (pdvstData->vstParameters[i].type == FLOAT_TYPE)
|
|
{
|
|
//beginEdit (i);
|
|
setParameterAutomated(i,
|
|
pdvstData->vstParameters[i].value.floatData);
|
|
//endEdit(i);
|
|
//l updateDisplay (); // I have commented this because it breaks sliders, numbox, etc. If this is on it jumps to max when first clicking on sliders, etc.
|
|
}
|
|
pdvstData->vstParameters[i].updated = 0;
|
|
}
|
|
}
|
|
|
|
if (pdvstData->guiName.direction == PD_SEND && \
|
|
pdvstData->guiName.updated)
|
|
{
|
|
if (pdvstData->guiName.type == STRING_TYPE)
|
|
{
|
|
strcpy(guiName, pdvstData->guiName.value.stringData);
|
|
|
|
pdvstData->guiName.updated=0;
|
|
// signal to vesteditor that guiname is updated
|
|
guiNameUpdated=true;
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
// to data chunk
|
|
|
|
if (pdvstData->datachunk.direction == PD_SEND && \
|
|
pdvstData->datachunk.updated)
|
|
{
|
|
if (pdvstData->datachunk.type = STRING_TYPE)
|
|
{
|
|
pdvstData->datachunk.updated=0;
|
|
}
|
|
}
|
|
|
|
ReleaseMutex(pdvstTransferMutex);
|
|
}
|
|
|
|
}
|
|
|
|
pdVstBuffer::pdVstBuffer(int nChans)
|
|
{
|
|
int i;
|
|
|
|
nChannels = nChans;
|
|
in = (float **)malloc(nChannels * sizeof(float *));
|
|
out = (float **)malloc(nChannels * sizeof(float *));
|
|
for (i = 0; i < nChannels; i++)
|
|
{
|
|
in[i] = (float *)calloc(DEFPDVSTBUFFERSIZE,
|
|
sizeof(float));
|
|
out[i] = (float *)calloc(DEFPDVSTBUFFERSIZE,
|
|
sizeof(float));
|
|
}
|
|
size = DEFPDVSTBUFFERSIZE;
|
|
}
|
|
|
|
pdVstBuffer::~pdVstBuffer()
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < nChannels; i++)
|
|
{
|
|
free(in[i]);
|
|
free(out[i]);
|
|
}
|
|
}
|
|
|
|
void pdVstBuffer::resize(int newSize)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < nChannels; i++)
|
|
{
|
|
in[i] = (float *)realloc(in[i], newSize * sizeof(float));
|
|
out[i] = (float *)realloc(out[i], newSize * sizeof(float));
|
|
}
|
|
size = newSize;
|
|
}
|
|
|
|
|