fluid_ladspa.c
上传用户:tjmskj2
上传日期:2020-08-17
资源大小:577k
文件大小:39k
源码类别:

midi

开发平台:

C/C++

  1. /* FluidSynth - A Software Synthesizer
  2.  *
  3.  * Copyright (C) 2003  Peter Hanappe and others.
  4.  *
  5.  * This library is free software; you can redistribute it and/or
  6.  * modify it under the terms of the GNU Library General Public License
  7.  * as published by the Free Software Foundation; either version 2 of
  8.  * the License, or (at your option) any later version.
  9.  *
  10.  * This library is distributed in the hope that it will be useful, but
  11.  * WITHOUT ANY WARRANTY; without even the implied warranty of
  12.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
  13.  * Library General Public License for more details.
  14.  *
  15.  * You should have received a copy of the GNU Library General Public
  16.  * License along with this library; if not, write to the Free
  17.  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
  18.  * 02111-1307, USA
  19.  */
  20. /* This module: 3/2002
  21.  * Author: Markus Nentwig, nentwig@users.sourceforge.net
  22.  */
  23. #define PrintErrorMessage -1
  24. #include "fluidsynth_priv.h"
  25. #ifdef LADSPA
  26. #include <assert.h>
  27. /* Dynamic library functions */
  28. #include <dlfcn.h>
  29. #include "fluid_ladspa.h"
  30. #include "fluid_synth.h"
  31. /* Logging to stdout. */
  32. //#define L(x) x;printf("n");
  33. #define L(x);
  34. fluid_LADSPA_FxUnit_t* new_fluid_LADSPA_FxUnit(fluid_synth_t* synth){
  35.   fluid_LADSPA_FxUnit_t* FxUnit=FLUID_NEW(fluid_LADSPA_FxUnit_t);
  36.   assert(FxUnit);
  37.   assert(synth);
  38.   /* The default state is 'bypassed'. The Fx unit has to be turned on explicitly by the user. */
  39.   /* Those settings have to be done in order to allow fluid_LADSPA_clean. */
  40.   FxUnit->Bypass=fluid_LADSPA_Bypassed;
  41.   FxUnit->NumberNodes=0;
  42.   FxUnit->NumberPlugins=0;
  43.   FxUnit->NumberLibs=0;
  44.   FxUnit->NumberCommands=0;
  45.   FxUnit->NumberUserControlNodes=0;
  46.   FxUnit->synth=synth;
  47.   pthread_cond_init(&FxUnit->cond,NULL);
  48.   return FxUnit;
  49. };
  50. /* Purpose:
  51.  * Creates the system nodes to get data into and out of the Fx unit.
  52.  */
  53. void fluid_LADSPA_CreateSystemNodes(fluid_LADSPA_FxUnit_t* FxUnit){
  54.   char str[99];
  55.   int nr_input_nodes;
  56.   int nr_fx_input_nodes;
  57.   int nr_output_nodes;
  58.   int temp;
  59.   int i;
  60.   /* Retrieve the number of synth / audio out / Fx send nodes */
  61.   assert(fluid_settings_getint(FxUnit->synth->settings, "synth.audio-groups", &temp));
  62.   nr_input_nodes=(int) temp;
  63.   printf("%i audio groupsn", nr_input_nodes);
  64.   assert(fluid_settings_getint(FxUnit->synth->settings, "synth.audio-channels", &temp));
  65.   nr_output_nodes=temp;
  66.   assert(fluid_settings_getint(FxUnit->synth->settings, "synth.effects-channels", &temp));
  67.   nr_fx_input_nodes=temp;
  68.   /* Create regular input nodes (associated with audio groups) */
  69.   for (i=0; i < nr_input_nodes; i++){
  70.       sprintf(str, "in%i_L",(i+1));
  71.       fluid_LADSPA_CreateNode(FxUnit, str, fluid_LADSPA_node_is_audio | fluid_LADSPA_node_is_source);
  72.       sprintf(str, "in%i_R",(i+1));
  73.       fluid_LADSPA_CreateNode(FxUnit, str, fluid_LADSPA_node_is_audio | fluid_LADSPA_node_is_source);
  74.   };
  75.   /* Create effects send nodes (for example reverb, chorus send) */
  76.   for (i=0; i < nr_fx_input_nodes; i++){
  77.       sprintf(str, "send%i_L",(i+1));
  78.       fluid_LADSPA_CreateNode(FxUnit, str, fluid_LADSPA_node_is_audio | fluid_LADSPA_node_is_source);
  79.       sprintf(str, "send%i_R",(i+1));
  80.       fluid_LADSPA_CreateNode(FxUnit, str, fluid_LADSPA_node_is_audio | fluid_LADSPA_node_is_source);
  81.   };
  82.   /* Create output nodes (usually towards the sound card) */
  83.   for (i=0; i < nr_input_nodes; i++){
  84.       sprintf(str, "out%i_L",(i+1));
  85.       fluid_LADSPA_CreateNode(FxUnit, str, fluid_LADSPA_node_is_audio | fluid_LADSPA_node_is_sink);
  86.       sprintf(str, "out%i_R",(i+1));
  87.       fluid_LADSPA_CreateNode(FxUnit, str, fluid_LADSPA_node_is_audio | fluid_LADSPA_node_is_sink);
  88.   };
  89. };
  90. /* Purpose:
  91.  * Creates predeclared nodes for control of the Fx unit during operation.
  92.  */
  93. void fluid_LADSPA_CreateUserControlNodes(fluid_LADSPA_FxUnit_t* FxUnit){
  94.   int i;
  95.   fluid_LADSPA_Node_t* CurrentNode;
  96.   for (i=0; i<FxUnit->NumberUserControlNodes; i++){
  97.     CurrentNode=fluid_LADSPA_CreateNode(FxUnit,FxUnit->UserControlNodeNames[i],fluid_LADSPA_node_is_control);
  98.     assert(CurrentNode);
  99.     CurrentNode->buf[0]=FxUnit->UserControlNodeValues[i];
  100.     CurrentNode->InCount++; /* The constant counts as input */
  101.     CurrentNode->flags=fluid_LADSPA_node_is_source | fluid_LADSPA_node_is_user_ctrl; /* It is a user control node */
  102.   };
  103. };
  104. /* Purpose:
  105.  * Returns the pointer to the shared library loaded using 'LibraryFilename' (if it has been loaded).
  106.  * Return NULL otherwise.
  107.  * If not: Clear Fx and abort.
  108.  */
  109. void * fluid_LADSPA_RetrieveSharedLibrary(fluid_LADSPA_FxUnit_t* FxUnit, char * LibraryFilename){
  110.   void * CurrentLib=NULL;
  111.   int LibCount;
  112.   for (LibCount=0; LibCount<FxUnit->NumberLibs; LibCount++){
  113.     assert(FxUnit->ppvPluginLibNames[LibCount]);
  114.     if (FLUID_STRCMP(FxUnit->ppvPluginLibNames[LibCount],LibraryFilename)==0){
  115.       CurrentLib=FxUnit->ppvPluginLibs[LibCount];
  116.     };
  117.   };
  118.   return CurrentLib;
  119. };
  120. /* Purpose:
  121.  * Loads a shared LADSPA library.
  122.  * Return NULL, if failed.
  123.  * TODO: use LADSPA_PATH
  124.  */
  125. void * fluid_LADSPA_LoadSharedLibrary(fluid_LADSPA_FxUnit_t* FxUnit, char * LibraryFilename){
  126.   void * LoadedLib;
  127.   assert(LibraryFilename);
  128.   LoadedLib=dlopen(LibraryFilename,RTLD_NOW);
  129.   if (!LoadedLib){
  130.     return NULL;
  131.   };
  132.   FxUnit->ppvPluginLibs[FxUnit->NumberLibs]=LoadedLib;
  133.   FxUnit->ppvPluginLibNames[FxUnit->NumberLibs]=FLUID_STRDUP(LibraryFilename);
  134.   FxUnit->NumberLibs++;
  135.   return LoadedLib;
  136. };
  137. /* Purpose:
  138.  * Retrieves a descriptor to the plugin labeled 'PluginLabel' from the shared library.
  139.  */
  140. const LADSPA_Descriptor * fluid_LADSPA_Retrieve_Plugin_Descriptor(void * CurrentLib, char * PluginLabel){
  141.   LADSPA_Descriptor_Function pfDescriptorFunction;
  142.   unsigned long lPluginIndex=0;
  143.   const LADSPA_Descriptor * psDescriptor;
  144.   pfDescriptorFunction = (LADSPA_Descriptor_Function)dlsym(CurrentLib,"ladspa_descriptor");
  145.   while (pfDescriptorFunction(lPluginIndex)){
  146.     psDescriptor = pfDescriptorFunction(lPluginIndex);
  147.     if (FLUID_STRCMP(psDescriptor->Label, PluginLabel) == 0){
  148.       return psDescriptor;
  149.     };
  150.     lPluginIndex++;
  151.   };
  152.   return NULL;
  153. };
  154. /* Purpose:
  155.  * Finds out, if 'PortName' starts with 'PluginPort'.
  156.  * Spaces and underscore mean the same.
  157.  * The comparison is not case sensitive.
  158.  * The result distinguishes between a full match (strings are equal), and a partial match (PortName starts with Plugin_Port, but is longer).
  159.  */
  160. fluid_LADSPA_Stringmatch_t fluid_LADSPA_Check_SubString_Match(const char * Plugin_Port, const char * PortName){
  161.   unsigned int CharCount;
  162.   char a;
  163.   char b;
  164.   for(CharCount=0; CharCount<FLUID_STRLEN(Plugin_Port); CharCount++){
  165.     a=PortName[CharCount];
  166.     b=Plugin_Port[CharCount];
  167.     if (a>='a' && a <='z'){a-=32;}
  168.     if (b>='a' && b <='z'){b-=32;}
  169.     if (a == ' '){a='_';};
  170.     if (b == ' '){b='_';};
  171.     if ( a != b){
  172.       return fluid_LADSPA_NoMatch;
  173.     };
  174.   };
  175.   if (FLUID_STRLEN(Plugin_Port) == FLUID_STRLEN(PortName)){
  176.     return fluid_LADSPA_FullMatch;
  177.   };
  178.   return fluid_LADSPA_PartialMatch;
  179. };
  180. /* Purpose:
  181.  * Load the plugins added with 'ladspa_add' and then start the Fx unit.
  182.  */
  183. int
  184. fluid_LADSPA_handle_start(fluid_synth_t* synth, int ac, char** av, fluid_ostream_t out){
  185.   fluid_LADSPA_FxUnit_t* FxUnit;
  186.   int CommandLineCount;
  187.   char * LibraryFilename;
  188.   char * PluginLabel;
  189.   char ** TokenSequence;
  190.   void * CurrentLib;
  191.   int NodeCount; //x
  192.   int IngoingSignalCount; // Count signals going into LADSPA Fx section
  193.   int OutgoingSignalCount; // Count signals going out of LADSPA Fx section
  194.   int ReturnVal=FLUID_OK; /* If warnings occur, this is set to -1. */
  195.   char * LADSPA_Path = getenv("LADSPA_PATH");
  196.   L(fluid_ostream_printf(out,"ladspa_start: starting..."));
  197.   assert(synth);
  198.   FxUnit=synth->LADSPA_FxUnit; assert(FxUnit);
  199.   /* When calling fluid_ladspastart, the Fx unit must be 'cleared' (no plugins, no libs, no nodes). Verify this here. */
  200.   if (FxUnit->NumberPlugins || FxUnit->NumberLibs){
  201.     fluid_ostream_printf(out, "***Error006***n"
  202.      "Fx unit is currently in use!n"
  203.      "Please run the ladspa_clear command before attempting to use ladspa_add!n");
  204.     /* In this case do _not_ clear the Fx unit. */
  205.     return(PrintErrorMessage);
  206.   };
  207.   if (!FxUnit->NumberCommands){
  208.     fluid_ostream_printf(out, "***Error007***n"
  209.      "Refusing to start the Fx unit without any plugin.n"
  210.      "Use ladspa_add first!n");
  211.     fluid_LADSPA_clear(FxUnit);
  212.     return(PrintErrorMessage);
  213.   };
  214.   /* Create predefined nodes */
  215.   L(fluid_ostream_printf(out,"ladspa_start: creating predefined nodes..."));
  216.   fluid_LADSPA_CreateSystemNodes(FxUnit);
  217.   /* Create predeclared nodes, that will allow to control the Fx unit during operation */
  218.   L(fluid_ostream_printf(out,"ladspa_start: creating user control nodes..."));
  219.   fluid_LADSPA_CreateUserControlNodes(FxUnit);
  220.   L(fluid_ostream_printf(out,"ladspa_start: Processing command lines..."));
  221.   for (CommandLineCount=0; CommandLineCount<FxUnit->NumberCommands; CommandLineCount++){
  222.     int CurrentPlugin_PortConnected[FLUID_LADSPA_MaxTokens/3]; /* For example: 100 tokens corresponds to roughly 30 ports. */
  223.     char LibFullPath[FLUID_LADSPA_MaxPathLength];
  224.     const LADSPA_Descriptor * CurrentPluginDescriptor;
  225.     LADSPA_Handle CurrentPlugin;
  226.     unsigned long PortCount;
  227.     int TokenCount=0;
  228.     char * Search;
  229.     int HasASlash=0;
  230.     if (FxUnit->NumberPlugins>=FLUID_LADSPA_MaxPlugins){
  231.     fluid_ostream_printf(out, "***Error003***n"
  232.        "Too many plugins at the same time (%i).n"
  233.        "Change FLUID_LADSPA_MaxPlugins!n",
  234.        FxUnit->NumberPlugins);
  235.       fluid_LADSPA_clear(FxUnit);
  236.       return(PrintErrorMessage);
  237.     };
  238.     L(fluid_ostream_printf(out,"Processing plugin nr. %i",FxUnit->NumberPlugins));
  239.     TokenSequence=FxUnit->LADSPA_Command_Sequence[CommandLineCount];
  240.     assert(TokenSequence);
  241.     /* Check, if the library is already loaded. If not, load. */
  242.     LibraryFilename=TokenSequence[TokenCount++]; assert(LibraryFilename);
  243.     L(fluid_ostream_printf(out,"Library name from ladspa_add: %s",LibraryFilename));
  244.     /* A slash-free filename refers to the LADSPA_PATH directory. Add that path, if needed. */
  245.     /* Determine, if the library filename contains a slash.
  246.      * If no, try to retrieve the environment variable LADSPA_PATH.
  247.      * If that fails, signal error.
  248.      * Otherwise leave the filename as it is.
  249.      */;
  250.     /* Determine, if the library name is just the filename, or a path (including a slash) */
  251.     Search=LibraryFilename;
  252.     while (*Search != '') {
  253.       if ((*Search)== '/'){
  254. HasASlash=1;
  255.       };
  256.       Search++;
  257.     };
  258.     if (!HasASlash){
  259.       if (!LADSPA_Path){
  260.     fluid_ostream_printf(out, "***Error018***n"
  261.  "The library file name %s does not include a path.n"
  262.  "The environment variable LADSPA_PATH is not set.n"
  263.  "- Use an absolute path (i.e. /home/myself/mylib.so)n"
  264.  "- For the current directory use ./mylib.son"
  265.  "- set the environment variable LADSPA_PATH (export LADSPA_PATH=/usr/lib/ladspa)n"
  266.  "- depending on your shell, try 'setenv' instead of 'export'n",
  267.  LibraryFilename);
  268. fluid_LADSPA_clear(FxUnit);
  269. return(PrintErrorMessage);
  270.       }; /* if no LADSPA_PATH */
  271.       snprintf(LibFullPath,FLUID_LADSPA_MaxPathLength,"%s/%s",LADSPA_Path,LibraryFilename);
  272.       /* If no slash in filename */
  273.     } else {
  274. snprintf(LibFullPath,FLUID_LADSPA_MaxPathLength,"%s",LibraryFilename);
  275.     };
  276.     L(fluid_ostream_printf(out,"Full Library path name: %s",LibFullPath));
  277.     CurrentLib=fluid_LADSPA_RetrieveSharedLibrary(FxUnit, LibFullPath);
  278.     if (!CurrentLib){
  279.       LADSPA_Descriptor_Function pfDescriptorFunction;
  280.       L(fluid_ostream_printf(out,"Library %s not yet loaded. Loading.",LibFullPath));
  281.       if (FxUnit->NumberLibs>=FLUID_LADSPA_MaxLibs){
  282.     fluid_ostream_printf(out, "***Error004***n"
  283.  "Too many libraries open (%i)n"
  284.  "Change FLUID_LADSPA_MaxLibs",FxUnit->NumberPlugins);
  285. fluid_LADSPA_clear(FxUnit);
  286. return(PrintErrorMessage);
  287.       };
  288.       /* Load a LADSPA plugin library and store it for future use.*/
  289.       CurrentLib=fluid_LADSPA_LoadSharedLibrary(FxUnit, LibFullPath);
  290.       if (!CurrentLib){
  291.     fluid_ostream_printf(out, "***Error008***n"
  292.  "Failed to load plugin library %s.",
  293.  LibraryFilename);
  294. fluid_LADSPA_clear(FxUnit);
  295. return(PrintErrorMessage);
  296.       };
  297.       dlerror();
  298.       pfDescriptorFunction = (LADSPA_Descriptor_Function)dlsym(CurrentLib,"ladspa_descriptor");
  299.       if (!pfDescriptorFunction) {
  300. const char * pcError = dlerror();
  301. if (!pcError) {pcError="Huh?! No error from lib!";};
  302.     fluid_ostream_printf(out, "***Error015***n"
  303.  "Unable to find ladspa_descriptor() function in plugin library file "%s": %s.n"
  304.  "Are you sure this is a LADSPA plugin file?n",
  305.  LibraryFilename,
  306.  pcError);
  307. fluid_LADSPA_clear(FxUnit);
  308. return(PrintErrorMessage);
  309.       };
  310.       L(fluid_ostream_printf(out,"Library loaded."));
  311.     };
  312.     PluginLabel=TokenSequence[TokenCount++]; assert(PluginLabel);
  313.     L(fluid_ostream_printf(out,"Plugin Label from ladspa_add: %s",PluginLabel));
  314.     /* Retrieve a 'plugin descriptor' from the library. */
  315.     L(fluid_ostream_printf(out,"Looking for the plugin labeled %s",PluginLabel));
  316.     CurrentPluginDescriptor=fluid_LADSPA_Retrieve_Plugin_Descriptor(CurrentLib, PluginLabel);
  317.     /* Check, that the plugin was actually found.*/
  318.     if (CurrentPluginDescriptor==NULL){
  319.     fluid_ostream_printf(out, "***Error016***n"
  320.        "Unable to find the plugin labeled "%s" in plugin library file "%s".n"
  321.        "Hint: run analyzeplugin %s from the command line to get a list of valid plugin labels.n",
  322.        PluginLabel,
  323.        LibraryFilename,
  324.        LibraryFilename);
  325.       fluid_LADSPA_clear(FxUnit);
  326.       return(PrintErrorMessage);
  327.     };
  328.     /* Create an instance of the plugin type from the descriptor */
  329.     L(fluid_ostream_printf(out,"instantiating plugin %s",PluginLabel));
  330.     CurrentPlugin=CurrentPluginDescriptor
  331.       ->instantiate(CurrentPluginDescriptor,44100); /* Sample rate hardcoded */
  332.     assert(CurrentPlugin);
  333.     /* The descriptor ("type of plugin") and the instance are stored for each plugin instantiation.
  334.        If one plugin type is instantiated several times, they will have the same descriptor, only
  335.        different instances.*/
  336.     FxUnit->PluginDescriptorTable[FxUnit->NumberPlugins]=CurrentPluginDescriptor;
  337.     FxUnit->PluginInstanceTable[FxUnit->NumberPlugins]=CurrentPlugin;
  338.     /*
  339.      *
  340.      * Wire up the inputs and outputs
  341.      *
  342.      */
  343.     /* List for checking, that each plugin port is exactly connected once */
  344.     for (PortCount=0; PortCount<CurrentPluginDescriptor->PortCount; PortCount++){
  345.       CurrentPlugin_PortConnected[PortCount]=0;
  346.     };
  347.     /* Note: There are three NULL tokens at the end. The last condition may be evaluated even if the first one already detects the end. */
  348.     while (TokenSequence[TokenCount] && TokenSequence[TokenCount+1] && TokenSequence[TokenCount+2]){
  349.       int CurrentPort_StringMatchCount;
  350.       LADSPA_PortDescriptor CurrentPort_Descriptor;
  351.       char * Plugin_Port=TokenSequence[TokenCount++];
  352.       char * Direction=TokenSequence[TokenCount++];
  353.       char * FLUID_Node=TokenSequence[TokenCount++];
  354.       fluid_LADSPA_Node_t* Current_Node;
  355.       const char * PortName=NULL;
  356.       int CurrentPort_Index=-1;
  357.       fluid_LADSPA_Stringmatch_t StringMatchType=fluid_LADSPA_NoMatch;
  358.       fluid_LADSPA_Stringmatch_t CurrentPort_StringMatchType=fluid_LADSPA_NoMatch;
  359.       CurrentPort_StringMatchCount=0;
  360.       L(fluid_ostream_printf(out,"Wiring %s %s %s",Plugin_Port, Direction, FLUID_Node));
  361.       /* Find the port number on the plugin, that belongs to "Plugin_Port".
  362.        * Match the identifier specified by the user against the first characters
  363.        * of the port name.
  364.        *
  365.        * If the given identifier matches several port names only partly, then the input is ambiguous, an error
  366.        * message results.
  367.        *
  368.        * Example: cmt.so, limit_peak: This plugin uses the labels
  369.        * -  Output Envelope Attack (s)
  370.        * -  Output Envelope Decay (s)
  371.        * -  Output
  372.        *
  373.        * The user input 'Output' matches the first two labels partly, the third fully. This will be accepted.
  374.        * The user input 'Out' matches all three only partly, this results in an error message.
  375.        */
  376.       for (PortCount=0; PortCount<CurrentPluginDescriptor->PortCount; PortCount++){
  377. PortName=CurrentPluginDescriptor->PortNames[PortCount];
  378. StringMatchType=fluid_LADSPA_Check_SubString_Match(Plugin_Port, PortName);
  379. /* If a full-string match has been found earlier, reject all partial matches. */
  380. if (StringMatchType==fluid_LADSPA_FullMatch ||
  381.     (StringMatchType==fluid_LADSPA_PartialMatch && CurrentPort_StringMatchType != fluid_LADSPA_FullMatch)){
  382.   if(StringMatchType==fluid_LADSPA_FullMatch && CurrentPort_StringMatchType==fluid_LADSPA_FullMatch){
  383.     fluid_ostream_printf(out, "***Error027***n"
  384.      "While processing plugin %s: The port label %s appears more than once!n"
  385.      "This is an error in the plugin itself. Please correct it or use another plugin.",PluginLabel, Plugin_Port);
  386.     fluid_LADSPA_clear(FxUnit);
  387.     return(PrintErrorMessage);
  388.   };
  389.   CurrentPort_Index=PortCount;
  390.   CurrentPort_StringMatchType=StringMatchType;
  391.   CurrentPort_StringMatchCount++;
  392. }; /* if suitable match */
  393.       }; /* For port count */
  394.       /* Several partial matches? Then the identifier is not unique. */
  395.       if (CurrentPort_StringMatchCount > 1 && CurrentPort_StringMatchType == fluid_LADSPA_PartialMatch){
  396.     fluid_ostream_printf(out, "***Error019***n"
  397.  "While processing plugin %s: The identifier %s matches more than one plugin port.n"
  398.  "Please use more letters for the port name. If needed, replace spaces with underscores (_).n"
  399.  "This error will not occur, if you use the full name of a port.n",PluginLabel, Plugin_Port);
  400. fluid_LADSPA_clear(FxUnit);
  401. return(PrintErrorMessage);
  402.       };
  403.       if (CurrentPort_Index<0){
  404.     fluid_ostream_printf(out, "***Error017***n"
  405.  "Unable to find port '%s' on plugin %sn"
  406.  "Port names are:n",Plugin_Port,PluginLabel);
  407. for (PortCount=0; PortCount<CurrentPluginDescriptor->PortCount; PortCount++){
  408.   printf("- `%s'n",CurrentPluginDescriptor->PortNames[PortCount]);
  409. };
  410. fluid_LADSPA_clear(FxUnit);
  411. return(PrintErrorMessage);
  412.       };
  413.       CurrentPort_Descriptor=CurrentPluginDescriptor->PortDescriptors[CurrentPort_Index];
  414.       assert(CurrentPort_Descriptor);
  415.       /* Retrieve the node with the right name. */
  416.       Current_Node=fluid_LADSPA_RetrieveNode(FxUnit,FLUID_Node);
  417. #define PortIsAudio LADSPA_IS_PORT_AUDIO(CurrentPort_Descriptor) && !(LADSPA_IS_PORT_CONTROL(CurrentPort_Descriptor))
  418. #define PortIsControl LADSPA_IS_PORT_CONTROL(CurrentPort_Descriptor) && !(LADSPA_IS_PORT_AUDIO(CurrentPort_Descriptor))
  419.       if (!Current_Node){
  420. /* Doesn't exist? Then create it. */
  421. if (FxUnit->NumberNodes>=FLUID_LADSPA_MaxNodes){
  422.     fluid_ostream_printf(out, "***Error005***n"
  423.    "Too many nodes (%i)n"
  424.    "Change FLUID_LADSPA_MaxNodes",FxUnit->NumberNodes);
  425.   fluid_LADSPA_clear(FxUnit);
  426.   return(PrintErrorMessage);
  427. };
  428. if (PortIsAudio){
  429.   Current_Node=fluid_LADSPA_CreateNode(FxUnit,FLUID_Node,fluid_LADSPA_node_is_audio);
  430. } else if (PortIsControl){
  431.   Current_Node=fluid_LADSPA_CreateNode(FxUnit,FLUID_Node,fluid_LADSPA_node_is_control);
  432. } else {
  433.     fluid_ostream_printf(out, "***Error025***n"
  434.    "Plugin port number %i is neither input nor output!n"
  435.    "This is an error in the plugin.n"
  436.    "Please check plugin sourcecode.n",
  437.    CurrentPort_Index);
  438.   fluid_LADSPA_clear(FxUnit);
  439.   return(PrintErrorMessage);
  440. };
  441.       };
  442.       assert(Current_Node);
  443.       /*
  444.        *
  445.        * Check flowgraph for some possible errors.
  446.        *
  447.        */
  448.       if (FLUID_STRCMP(Direction,"->")==0){
  449. /* Data from plugin to FLUID, into node
  450.  *
  451.  * *** Rule: ****
  452.  * A node may not have more than one data source.*/
  453. if (Current_Node->InCount !=0){
  454.     fluid_ostream_printf(out, "***Error009***n"
  455.    "Plugin %s tries to feed data from output %s into node %s, which is already connected to a data source.n",PluginLabel,Plugin_Port,FLUID_Node);
  456.   fluid_LADSPA_clear(FxUnit);
  457.   return(PrintErrorMessage);
  458. };
  459. Current_Node->InCount++;
  460.       } else if (FLUID_STRCMP(Direction,"<-")==0){
  461. /* Data from FLUID to plugin, out of node
  462.  *
  463.  * This check verifies the integrity of the flow graph:
  464.  * *** Rule ***
  465.  * The execution order of the plugins is the order, in which they are programmed.
  466.  * The plugins must be ordered so, that the input of a plugin is already computed at the time of its execution.
  467.  * If the user tries to read data out of a node that has not yet an input, then something is wrong.*/
  468. assert(Current_Node->InCount<=1);
  469. if (Current_Node->InCount !=1){
  470.     fluid_ostream_printf(out, "***Error010***n"
  471.    "Plugin %s tries to read data through input %s from node %s.n"
  472.    "But at this point there is no valid data at that node.n"
  473.    "Please check the flowgraph and especially the execution order!n",PluginLabel,Plugin_Port,FLUID_Node);
  474.   fluid_LADSPA_clear(FxUnit);
  475.   return(PrintErrorMessage);
  476. };
  477. Current_Node->OutCount++;
  478.       } else {
  479. fluid_ostream_printf(out, "***Error024***n"
  480.        "Syntax error: Illegal `arrow' `%s', expecting -> or <-n",
  481.        Direction);
  482. fluid_LADSPA_clear(FxUnit);
  483. return(PrintErrorMessage);
  484.       };
  485.       /* In any case, there must be a valid data source for the port at this time. */
  486.       assert(Current_Node->InCount==1);
  487.       /* Keep track on the number of connections to each port.
  488.        * This error occurs only, if an attempt is made to connect one port twice (i.e. ladspa_add libname pluginname port <- nodex port <- nodey) */
  489.       if (CurrentPlugin_PortConnected[CurrentPort_Index]){
  490.     fluid_ostream_printf(out, "***Error011***n"
  491.  "Refusing to connect twice to port %s on plugin %s.n",CurrentPluginDescriptor->PortNames[CurrentPort_Index], PluginLabel);
  492. fluid_LADSPA_clear(FxUnit);
  493. return(PrintErrorMessage);
  494.       };
  495.       /*
  496.        *
  497.        * Connect the port
  498.        *
  499.        */
  500.       L(fluid_ostream_printf(out,"Connecting %i",CurrentPort_Index));
  501.       CurrentPluginDescriptor->connect_port
  502. (CurrentPlugin,
  503.  CurrentPort_Index,
  504.  Current_Node->buf
  505.  );
  506.       CurrentPlugin_PortConnected[CurrentPort_Index]++;
  507.     }; /* While Tokensequence (more connections) */
  508.     /*
  509.      *
  510.      * Check for left-over tokens
  511.      *
  512.      */
  513.     if (TokenSequence[TokenCount]){
  514.       char * T1="";char * T2="";char * T3="";
  515.       if (TokenSequence[TokenCount]){T1=TokenSequence[TokenCount];};
  516.       if (TokenSequence[TokenCount+1]){T2=TokenSequence[TokenCount+1];};
  517.       if (TokenSequence[TokenCount+2]){T3=TokenSequence[TokenCount+2];};
  518.     fluid_ostream_printf(out, "***Error012***n"
  519.        "Leftover tokens: %s %s %s...n",T1,T2,T3);
  520.       fluid_LADSPA_clear(FxUnit);
  521.       return(PrintErrorMessage);
  522.     };
  523.     /*
  524.      *
  525.      * Check, that all plugin ports are connected
  526.      *
  527.      */
  528.     L(fluid_ostream_printf(out,"Checking left-over ports"));
  529.     assert(CurrentPluginDescriptor);
  530.     for (PortCount=0; PortCount<CurrentPluginDescriptor->PortCount; PortCount++){
  531.       assert(CurrentPlugin_PortConnected[PortCount] <=1);
  532.       if (CurrentPlugin_PortConnected[PortCount] !=1){
  533.     fluid_ostream_printf(out, "***Error013***nPlugin: %s. Port %s is unconnected!n",PluginLabel, CurrentPluginDescriptor->PortNames[PortCount]);
  534. fluid_LADSPA_clear(FxUnit);
  535. return(PrintErrorMessage);
  536.       };
  537.     };
  538.     /*
  539.      *
  540.      *Run activate function on plugin, where possible
  541.      *
  542.      */
  543.     if (CurrentPluginDescriptor->activate !=NULL){
  544.       CurrentPluginDescriptor->activate(CurrentPlugin);
  545.     };
  546.     FxUnit->NumberPlugins++;
  547.   }; /* For CommandLineCount: once for each new command (i.e. plugin instantiation request from user */
  548.   /*
  549.    *
  550.    * Further flow graph checks
  551.    *
  552.    */
  553.   L(fluid_ostream_printf(out,"Checking flow graph"));
  554.   IngoingSignalCount=0;
  555.   OutgoingSignalCount=0;
  556.   for (NodeCount=0; NodeCount<FxUnit->NumberNodes; NodeCount++){
  557.     fluid_LADSPA_Node_t* Current_Node;
  558.     Current_Node=FxUnit->Nodelist[NodeCount];
  559.     assert(Current_Node);
  560.     if (Current_Node->flags & fluid_LADSPA_node_is_source){
  561.       IngoingSignalCount+=Current_Node->OutCount;
  562.     } else if (Current_Node->flags & fluid_LADSPA_node_is_sink){
  563.       OutgoingSignalCount+=Current_Node->InCount;
  564.     } else {
  565.       /* A node without any input doesn't make sense.
  566.        * The flow graph check aborts with an error. This case cannot happen.
  567.        */
  568.       if (Current_Node->InCount==0 && !Current_Node->flags & fluid_LADSPA_node_is_dummy){ /* There can only be one warning at a time. */
  569.     fluid_ostream_printf(out, "***Warning020***"
  570.    "No input into node %s.n"
  571.    "Use '_' as first char in nodename to suppress this warning.n"
  572.    "Hint: Check for typos in the node name.n",Current_Node->Name);
  573.   /* A warning can also be printed as an error message (check fluid_cmd.c). The only difference between
  574.      the return values -1 and 0 is, that -1 prints the result. */
  575.       };
  576.       ReturnVal=PrintErrorMessage;
  577.     };
  578.     /* A node without any output doesn't make sense. */
  579.     if (Current_Node->OutCount==0 && !Current_Node->flags & fluid_LADSPA_node_is_dummy){
  580.     fluid_ostream_printf(out, "***Warning021***n"
  581.  "No output from node %s.n"
  582.  "Use '_' as first char in nodename to suppress this warning.n"
  583.  "Hint: Check for typos in the node name.n",Current_Node->Name);
  584.       ReturnVal=PrintErrorMessage;
  585.     };
  586.     /* A free-flying node simply cannot happen. */
  587.     assert(Current_Node->OutCount+Current_Node->InCount);
  588.   }; /* Foreach node */
  589.   /* Issue a warning, if no signal goes into the Fx section. */
  590.   if (IngoingSignalCount==0){
  591.     fluid_ostream_printf(out, "***Warning022***n"
  592.        "You have not connected anything to the synthesizer section (in1_L, in1_R).n");
  593.     ReturnVal=PrintErrorMessage;
  594.   };
  595.   /* Issue a warning, if no signal leaves the Fx section. */
  596.   if (OutgoingSignalCount==0){
  597.     fluid_ostream_printf(out, "***Warning023***n"
  598.        "You have not connected anything to the output (out1_L, out1_R).n");
  599.   };
  600.   /* Finally turn on the Fx unit. */
  601.   FxUnit->Bypass=fluid_LADSPA_Active;
  602.   L(fluid_ostream_printf(out,"LADSPA Init OK"));
  603.   return(ReturnVal);
  604. };
  605. void
  606. fluid_LADSPA_run(fluid_LADSPA_FxUnit_t* FxUnit, fluid_real_t* left_buf[], fluid_real_t* right_buf[], fluid_real_t* fx_left_buf[], fluid_real_t* fx_right_buf[]){
  607.   int i;
  608.   int ii;
  609.   int nr_audio_channels;
  610.   int nr_fx_sends;
  611.   int nr_groups;
  612.   int byte_size = FLUID_BUFSIZE * sizeof(fluid_real_t);
  613.   char str[99];
  614.   fluid_LADSPA_Node_t* n;
  615.   int temp;
  616.   /* Retrieve the number of synth / audio out / Fx send nodes */
  617.   assert(fluid_settings_getint(FxUnit->synth->settings, "synth.audio-groups", &temp));
  618.   nr_groups=(int) temp;
  619.   assert(fluid_settings_getint(FxUnit->synth->settings, "synth.audio-channels", &temp));
  620.   nr_audio_channels=temp;
  621.   assert(fluid_settings_getint(FxUnit->synth->settings, "synth.effects-channels", &temp));
  622.   nr_fx_sends=temp;
  623.   /* Fixme: Retrieving nodes via names is inefficient
  624.    * (but not that bad, because the interesting nodes are always at the start of the list).
  625.    */
  626.   /* Input and output are processed via the same buffers. Therefore the effect is bypassed by just skipping everything else. */
  627.   if (FxUnit->Bypass==fluid_LADSPA_Bypassed){
  628.     return;
  629.   };
  630.   if (FxUnit->Bypass==fluid_LADSPA_BypassRequest){
  631.     FxUnit->Bypass=fluid_LADSPA_Bypassed;
  632.     pthread_mutex_lock(&FxUnit->mutex);
  633.     pthread_cond_broadcast(&FxUnit->cond);
  634.     pthread_mutex_unlock(&FxUnit->mutex);
  635.     L(printf("LADSPA_Run: Command line asked for bypass of Fx unit. Acknowledged."));
  636.     return;
  637.   };
  638.   assert(FxUnit);
  639.   /* Clear the output buffers, if they are not connected anywhere */
  640.   for (ii=0; ii < nr_audio_channels; ii++){
  641.       sprintf(str, "out%i_L",(ii+1));
  642.       n=fluid_LADSPA_RetrieveNode(FxUnit, str); assert(n);
  643.       if (n->InCount == 0){
  644. /*   printf("Output node %s is not connected -> clear buffern", str); */
  645.   FLUID_MEMSET(n->buf, 0, byte_size);
  646.       };
  647.       sprintf(str, "out%i_R",(ii+1));
  648.       n=fluid_LADSPA_RetrieveNode(FxUnit, str); assert(n);
  649.       if (n->InCount == 0){
  650. /*   printf("Output node %s is not connected -> clear buffern", str); */
  651.   FLUID_MEMSET(n->buf, 0, byte_size);
  652.       };
  653.   };
  654.   /* Prepare the incoming data:
  655.    * Convert fluid_real_t data type to LADSPA_Data type */
  656.   for (ii=0; ii < nr_groups; ii++){
  657.       fluid_real_t* src_buf=left_buf[ii];
  658.       sprintf(str, "in%i_L",(ii+1));
  659.       n=fluid_LADSPA_RetrieveNode(FxUnit, str); assert(n);
  660.       assert(FLUID_BUFSIZE % 2 == 0);
  661.       /* Add a very small high frequency signal. This avoids denormal number problems. */
  662.       for (i=0; i<FLUID_BUFSIZE;){
  663.   n->buf[i]=(LADSPA_Data)(src_buf[i]+1.e-15);
  664.   i++;
  665.   n->buf[i]=(LADSPA_Data)(src_buf[i]);
  666.   i++;
  667.       };
  668.       src_buf=right_buf[ii];
  669.       sprintf(str, "in%i_R",(ii+1));
  670.       n=fluid_LADSPA_RetrieveNode(FxUnit, str); assert(n);
  671.       /* Add a very small high frequency signal. This avoids denormal number problems. */
  672.       for (i=0; i<FLUID_BUFSIZE;){
  673.   n->buf[i]=(LADSPA_Data)(src_buf[i]+1.e-15);
  674.   i++;
  675.   n->buf[i]=(LADSPA_Data)(src_buf[i]);
  676.   i++;
  677.       };
  678.   };
  679.   /* Effect send paths */
  680.   for (ii=0; ii < nr_fx_sends; ii++){
  681.       sprintf(str, "send%i_L",(ii+1));
  682.       n=fluid_LADSPA_RetrieveNode(FxUnit, str); assert(n);
  683.       for (i=0; i<FLUID_BUFSIZE; i++){
  684.   n->buf[i]=(LADSPA_Data)(fx_left_buf[ii][i]);
  685.       };
  686.       sprintf(str, "send%i_R",(ii+1));
  687.       n=fluid_LADSPA_RetrieveNode(FxUnit, str); assert(n);
  688.       for (i=0; i<FLUID_BUFSIZE; i++){
  689.   n->buf[i]=(LADSPA_Data)(fx_right_buf[ii][i]);
  690.       };
  691.   };
  692.   /* Run each plugin on a block of data.
  693.    * The execution order has been checked during setup.*/
  694.   for (i=0; i<FxUnit->NumberPlugins; i++){
  695.     FxUnit->PluginDescriptorTable[i]->run(FxUnit->PluginInstanceTable[i],FLUID_BUFSIZE);
  696.   };
  697.   /* Copy the data from the output nodes back to the synth. */
  698.   for (ii=0; ii < nr_audio_channels; ii++){
  699.       fluid_real_t* dest_buf=left_buf[ii];
  700.       sprintf(str, "out%i_L",(ii+1));
  701.       n=fluid_LADSPA_RetrieveNode(FxUnit, str); assert(n);
  702.       for (i=0; i<FLUID_BUFSIZE; i++){
  703.   dest_buf[i]=(fluid_real_t)n->buf[i];
  704.       };
  705.       dest_buf=right_buf[ii];
  706.       sprintf(str, "out%i_R",(ii+1));
  707.       n=fluid_LADSPA_RetrieveNode(FxUnit, str); assert(n);
  708.       for (i=0; i<FLUID_BUFSIZE; i++){
  709.   dest_buf[i]=(fluid_real_t)n->buf[i];
  710.       };
  711.   };
  712. };
  713. fluid_LADSPA_Node_t*
  714. fluid_LADSPA_RetrieveNode(fluid_LADSPA_FxUnit_t* FxUnit, char * Name){
  715.   int i=0;
  716.   assert(FxUnit);assert(Name);
  717.   for (i=0; i<FxUnit->NumberNodes; i++){
  718.     assert(FxUnit->Nodelist[i]);
  719.     if (FLUID_STRCMP(FxUnit->Nodelist[i]->Name,Name)==0){
  720.       return FxUnit->Nodelist[i];
  721.     };
  722.   };
  723.   return NULL;
  724. };
  725. /* Purpose:
  726.  * Creates a new node from the node name given by the user.
  727.  */
  728. fluid_LADSPA_Node_t*
  729. fluid_LADSPA_CreateNode(fluid_LADSPA_FxUnit_t* FxUnit, char * Name, int flags){
  730.   int Dummy=0;
  731.   fluid_LADSPA_Node_t* NewNode;
  732.   assert(FxUnit);
  733.   assert(Name);
  734. //  printf("Flags is %in",flags);
  735.   L(printf("Create node: %s",Name));
  736.   if (FxUnit->NumberNodes>=FLUID_LADSPA_MaxNodes){
  737.     printf( "***Error014***n"
  738.     "Too many nodes (%i)n"
  739.     "Change FLUID_LADSPA_MaxNodes.n",FxUnit->NumberNodes);
  740.     fluid_LADSPA_clear(FxUnit);
  741.     return NULL;
  742.   };
  743.   /* Don't allow node names, which start with -, 0..9 */
  744.   if (Name[0] == '-' || (Name[0]>='0' && Name[0]<='9')){
  745.     printf( "***Error026***n"
  746.     "The node name %s starts with a digit / minus sign!n"
  747.     "Please use a letter to start a node name.n"
  748.     "A constant node is created by using `#' as first character,n"
  749.     "for example #-2.5.n",
  750.     Name);
  751.     fluid_LADSPA_clear(FxUnit);
  752.     return NULL;
  753.   };
  754.   /* A nodename starting with "_" is a possible dummy node, which may (but need not) act as a data sink or source (dummy node). */
  755.   if (Name[0] == ' '){ /* ??? Should be '_' ??? */
  756.     Dummy=1;
  757.   };
  758.   NewNode=FLUID_NEW(fluid_LADSPA_Node_t);assert(NewNode);
  759.   if (flags && fluid_LADSPA_node_is_audio){
  760.     /* Audio node contains buffer. */
  761.     NewNode->buf=FLUID_ARRAY(LADSPA_Data, (FLUID_BUFSIZE));assert(NewNode->buf);
  762.     /* It is permitted to use a dummy node without input. Therefore clear all node buffers at startup. */
  763.     FLUID_MEMSET(NewNode->buf, 0, (FLUID_BUFSIZE*sizeof(LADSPA_Data)));
  764.   } else if (flags & fluid_LADSPA_node_is_control){
  765.     /* Control node contains single value. */
  766.     NewNode->buf=FLUID_ARRAY(LADSPA_Data, 1);assert(NewNode->buf);
  767.   } else {
  768.     assert(0);
  769.   };
  770.   NewNode->Name=FLUID_STRDUP(Name);assert(NewNode->Name);
  771.   if (Dummy){
  772.       flags |= fluid_LADSPA_node_is_dummy;
  773.   };
  774.   NewNode->InCount=0;
  775.   NewNode->OutCount=0;
  776.   NewNode->flags=flags;
  777.   /* A nodename starting with "#" means that the node holds a constant value. */
  778.   if (NewNode->Name[0] == '#'){
  779.     assert(flags & fluid_LADSPA_node_is_control);
  780.     /* Skip the first character => +1 */
  781.     NewNode->buf[0]=(LADSPA_Data)atof(NewNode->Name+1);
  782.     NewNode->InCount++;
  783.   };
  784.   if (flags & fluid_LADSPA_node_is_source){
  785.     NewNode->InCount++;
  786. //    printf("****************************** Source!n");
  787.   } else if (flags & fluid_LADSPA_node_is_sink){
  788.     NewNode->OutCount++;
  789.   };
  790.   FxUnit->Nodelist[FxUnit->NumberNodes++]=NewNode;
  791.   L(printf("Node %s created.",Name));
  792.   return NewNode;
  793. };
  794. void fluid_LADSPA_clear(fluid_LADSPA_FxUnit_t* FxUnit){
  795.   int i;
  796.   int ii;
  797.   L(printf("ladspa_clear"));
  798.   assert(FxUnit);
  799.   if (FxUnit->Bypass==fluid_LADSPA_Active){
  800.     L(printf("clear: Requesting bypass from synthesis thread"));
  801.     /* Bypass the Fx unit before anything else.
  802.      * Reason: Not a good idea to release plugins, while another thread runs them.
  803.      */
  804.     FxUnit->Bypass=fluid_LADSPA_BypassRequest;
  805.     pthread_mutex_lock(&FxUnit->mutex);
  806.     pthread_cond_wait(&FxUnit->cond,&FxUnit->mutex);
  807.     pthread_mutex_unlock(&FxUnit->mutex);
  808.     L(printf("clear: Synthesis thread has switched to bypass."));
  809.   } else {
  810.     L(printf("clear: Fx unit was already bypassed. No action needed."));
  811.   };
  812.   L(printf("Clear all user control node declarations"));
  813.   for (i=0; i<FxUnit->NumberUserControlNodes; i++){
  814.     FLUID_FREE(FxUnit->UserControlNodeNames[i]);
  815.   };
  816.   FxUnit->NumberUserControlNodes=0;
  817.   L(printf("Clear all plugin instances"));
  818.   for (i=0; i<FxUnit->NumberPlugins; i++){
  819.     assert(FxUnit->PluginDescriptorTable[i]);
  820.     assert(FxUnit->PluginInstanceTable[i]);
  821.     /* Run deactivate function on plugin, if possible */
  822.     if (FxUnit->PluginDescriptorTable[i]->deactivate){
  823.       FxUnit->PluginDescriptorTable[i]->deactivate(FxUnit->PluginInstanceTable[i]);
  824.     };
  825.     FxUnit->PluginDescriptorTable[i]->cleanup(FxUnit->PluginInstanceTable[i]);
  826.   };
  827.   FxUnit->NumberPlugins=0;
  828.   L(printf("Clear all nodes")); /* Only after removing plugins! */
  829.   for (i=0; i<FxUnit->NumberNodes; i++){
  830.     FLUID_FREE(FxUnit->Nodelist[i]->buf);
  831.     FLUID_FREE(FxUnit->Nodelist[i]);
  832.   };
  833.   FxUnit->NumberNodes=0;
  834.   L(printf("Clear all plugin libraries"));
  835.   for (i=0; i<FxUnit->NumberLibs; i++){
  836.     assert(FxUnit->ppvPluginLibs[i]);
  837.     dlclose(FxUnit->ppvPluginLibs[i]);
  838.     assert(FxUnit->ppvPluginLibNames[i]);
  839.     FLUID_FREE(FxUnit->ppvPluginLibNames[i]);
  840.   };
  841.   FxUnit->NumberLibs=0;
  842.   L(printf("Clear all command lines"));
  843.   for (i=0; i<FxUnit->NumberCommands;i++){
  844.     ii=0;
  845.     assert(FxUnit->LADSPA_Command_Sequence[i]);
  846.     while (FxUnit->LADSPA_Command_Sequence[i][ii]){
  847.       FLUID_FREE(FxUnit->LADSPA_Command_Sequence[i][ii]);
  848.       ii++;
  849.     };
  850.     FLUID_FREE(FxUnit->LADSPA_Command_Sequence[i]);
  851.   };
  852.   FxUnit->NumberCommands=0;
  853. };
  854. int fluid_LADSPA_handle_add(fluid_synth_t* synth, int ac, char** av, fluid_ostream_t out){
  855.   int i;
  856.   char * Token;
  857.   char ** CommandLine;
  858.   fluid_LADSPA_FxUnit_t* FxUnit;
  859.   assert(synth);
  860.   FxUnit=synth->LADSPA_FxUnit; assert(FxUnit);
  861.   if (ac>=FLUID_LADSPA_MaxTokens){
  862.     /* Can't be tested. fluidsynth limits the number of tokens. */
  863.     printf("***Error001***n"
  864.      "Too many ports.nChange FLUID_LADSPA_MaxTokens!");
  865.     fluid_LADSPA_clear(FxUnit);
  866.     return(PrintErrorMessage);
  867.   };
  868.   if (ac<2){
  869.     printf("***Error002***n"
  870.      "ladspa_add needs at least two arguments - libname and plugin name!");
  871.     fluid_LADSPA_clear(FxUnit);
  872.     return(PrintErrorMessage);
  873.   };
  874.   if (FxUnit->NumberCommands>=FLUID_LADSPA_MaxPlugins){
  875.     printf("***Error032***n"
  876.      "Too many plugins.nChange FLUID_LADSPA_MaxPlugins!");
  877.     fluid_LADSPA_clear(FxUnit);
  878.     return(PrintErrorMessage);
  879.   };
  880.   /* CommandLine (token sequence) is terminated with NULL.
  881.    * Add two more NULLs, so that a chunk of three tokens can be checked later without risk.*/
  882.   CommandLine=FLUID_ARRAY(char*, (ac+3));assert(CommandLine);
  883.   for (i=0; i<ac; i++){
  884.     CommandLine[i]=FLUID_STRDUP(av[i]);assert(CommandLine[i]);
  885.   };
  886.   CommandLine[ac]=NULL;
  887.   CommandLine[ac+1]=NULL;
  888.   CommandLine[ac+2]=NULL;
  889.   FxUnit->LADSPA_Command_Sequence[FxUnit->NumberCommands]=CommandLine;
  890.   FxUnit->NumberCommands++;
  891.   return(FLUID_OK);
  892. };
  893. int fluid_LADSPA_handle_declnode(fluid_synth_t* synth, int ac, char** av, fluid_ostream_t out){
  894.   int i;
  895.   char * Token;
  896.   char ** CommandLine;
  897.   char * NodeName;
  898.   fluid_real_t NodeValue;
  899.   fluid_LADSPA_FxUnit_t* FxUnit;
  900.   assert(synth);
  901.   FxUnit=synth->LADSPA_FxUnit; assert(FxUnit);
  902.   if (ac<2){
  903.     printf("***Error028***n"
  904.    "ladspa_declnode needs two arguments - node name and value!n");
  905.     fluid_LADSPA_clear(FxUnit);
  906.     return(PrintErrorMessage);
  907.   };
  908.   if (FxUnit->NumberUserControlNodes>=FLUID_LADSPA_MaxNodes){
  909.     printf("***Error033***n"
  910.      "Too many user-control nodes.nChange FLUID_LADSPA_MaxNodes!");
  911.     fluid_LADSPA_clear(FxUnit);
  912.     return(PrintErrorMessage);
  913.   };
  914.   NodeName=FLUID_STRDUP(av[0]); assert(NodeName);
  915.   NodeValue=atof(av[1]);
  916.   FxUnit->UserControlNodeNames[FxUnit->NumberUserControlNodes]=NodeName;
  917.   FxUnit->UserControlNodeValues[FxUnit->NumberUserControlNodes]=NodeValue;
  918.   FxUnit->NumberUserControlNodes++;
  919.   return(FLUID_OK);
  920. };
  921. int fluid_LADSPA_handle_setnode(fluid_synth_t* synth, int ac, char** av, fluid_ostream_t out){
  922.   int i;
  923.   char * Token;
  924.   char * NodeName;
  925.   fluid_real_t NodeValue;
  926.   fluid_LADSPA_FxUnit_t* FxUnit;
  927.   fluid_LADSPA_Node_t* CurrentNode;
  928.   assert(synth);
  929.   FxUnit=synth->LADSPA_FxUnit; assert(FxUnit);
  930.   if (ac!=2){
  931.     printf("***Error029***n"
  932.    "ladspa_setnode needs two arguments - node name and value!n");
  933.     /* Do not clear the Fx unit (no fluid_LADSPA_clear). */
  934.     return(PrintErrorMessage);
  935.   };
  936.   NodeName=av[0]; assert(NodeName);
  937.   NodeValue=atof(av[1]);
  938.   CurrentNode=fluid_LADSPA_RetrieveNode(FxUnit,NodeName);
  939.   if (!CurrentNode){
  940.     printf("***Error030***n"
  941.    "The node %s was not found. Please use the full name of a node, that wasn"
  942.    "previously declared with ladspa_declnode.n",NodeName);
  943.     /* Do not clear the Fx unit (no fluid_LADSPA_clear). */
  944.     return(PrintErrorMessage);
  945.   };
  946.   if (!(CurrentNode->flags & fluid_LADSPA_node_is_user_ctrl)){
  947.     printf("***Error031***n"
  948.    "The node %s is an ordinary control node.n"
  949.    "Only user control nodes can be modified with ladspa_setnode.n",NodeName);
  950.     /* Do not clear the Fx unit (no fluid_LADSPA_clear). */
  951.     return(PrintErrorMessage);
  952.   };
  953.   L(printf("ladspa_setnode: Assigning value %f",NodeValue));
  954.   CurrentNode->buf[0]=NodeValue;
  955.   return(FLUID_OK);
  956. };
  957. int fluid_LADSPA_handle_clear(fluid_synth_t* synth, int ac, char** av, fluid_ostream_t out){
  958.   fluid_LADSPA_FxUnit_t* FxUnit;
  959.   assert(synth);
  960.   FxUnit=synth->LADSPA_FxUnit; assert(FxUnit);
  961.   fluid_LADSPA_clear(FxUnit);
  962.   return(FLUID_OK);
  963. };
  964. void fluid_LADSPA_shutdown(fluid_LADSPA_FxUnit_t* FxUnit){
  965.   /* The synthesis thread is not running anymore.
  966.    * Set the bypass switch, so that fluid_LADSPA_clear can proceed.*/
  967.   FxUnit->Bypass=fluid_LADSPA_Bypassed;
  968.   fluid_LADSPA_clear(FxUnit);
  969.   pthread_cond_destroy(&FxUnit->cond); /* pro forma */
  970. };
  971. #endif /*LADSPA*/