From 772072ec8d13ad31eba456a01873b84577f3f2da Mon Sep 17 00:00:00 2001
From: Albert Graef <aggraef@gmail.com>
Date: Sun, 16 Aug 2020 21:06:49 +0200
Subject: [PATCH] Improve bendin compatibility with vanilla.

This adds an option to make bendin use the vanilla range of 0..16383 for
compatibility. As was observed a long time ago, this is at odds with the
bendout range of -8192..8191, and was thus fixed in pd-l2ork at some
point. But Miller Puckette has decided that this bug won't *ever* be fixed in
vanilla, and this is even documented in vanilla's midi-help.pd patch.

This discrepancy breaks a lot of patches involving MIDI, so a compatibility
option is needed. It's been possible to use -legacy to get vanilla's bendin,
but it's better not to conflate legacy iemgui positioning issues with the
bendin implementation. The former won't usually break interoperability, but
the latter does.

So, as suggested by Ico Bukvic, we add a flag as an optional second argument
to bendin instead. If present, this option explicitly denotes the target
range: zero means the default (pd-l2ork, signed) output range, nonzero the
vanilla-compatible unsigned range. In addition, there's a new -legacy-bendin
command line option which makes unsigned output the default for bendin objects
which don't specify the range explicitly. You'll want to put this into your
startup flags in the preferences if you need out-of-the-box vanilla
compatibility.
---
 pd/src/s_main.c |  9 +++++++++
 pd/src/x_midi.c | 21 ++++++++++++---------
 2 files changed, 21 insertions(+), 9 deletions(-)

diff --git a/pd/src/s_main.c b/pd/src/s_main.c
index 62ad4dc69..f3c3d14ac 100644
--- a/pd/src/s_main.c
+++ b/pd/src/s_main.c
@@ -57,6 +57,8 @@ int sys_unique = 0;     /* by default off, prevents multiple instances
                            of pd-l2ork */
 int sys_legacy = 0;     /* by default off, used to enable legacy features,
                            such as offsets in iemgui object positioning */
+int sys_legacy_bendin = 0; /* by default off, used to enable vanilla-
+                              compatible (unsigned) pitch bend input */
 char *sys_guicmd;
 t_symbol *sys_gui_preset; /* name of gui theme to be used */
 t_symbol *sys_libdir;
@@ -531,6 +533,7 @@ static char *(usagemessage[]) = {
 "-k12             -- enable K-12 education mode (requires L2Ork K12 lib)\n",
 "-unique          -- enable multiple instances (disabled by default)\n",
 "-legacy          -- enable legacy features (disabled by default)\n", 
+"-legacy-bendin   -- enable legacy (unsigned) bendin (disabled by default)\n", 
 "\n",
 };
 
@@ -938,6 +941,12 @@ int sys_argparse(int argc, char **argv)
             argc -= 1;
             argv += 1;
         }
+        else if (!strcmp(*argv, "-legacy-bendin"))
+        {
+            sys_legacy_bendin = 1;
+            argc -= 1;
+            argv += 1;
+        }
         else if (!strcmp(*argv, "-guiport") && argc > 1 &&
             sscanf(argv[1], "%d", &sys_guisetportnumber) >= 1)
         {
diff --git a/pd/src/x_midi.c b/pd/src/x_midi.c
index c827ea737..2f75442d7 100644
--- a/pd/src/x_midi.c
+++ b/pd/src/x_midi.c
@@ -307,14 +307,21 @@ typedef struct _bendin
 {
     t_object x_obj;
     t_float x_channel;
+    t_float x_offs;
     t_outlet *x_outlet1;
     t_outlet *x_outlet2;
 } t_bendin;
 
-static void *bendin_new(t_floatarg f)
+static void *bendin_new(t_symbol *s, int argc, t_atom *argv)
 {
     t_bendin *x = (t_bendin *)pd_new(bendin_class);
+    extern int sys_legacy_bendin;
+    t_float f = 0.0, g = sys_legacy_bendin;
+    f = atom_getfloatarg(0, argc, argv);
+    if (argc > 1)
+      g = atom_getfloatarg(1, argc, argv);
     x->x_channel = f;
+    x->x_offs = g==0.0?-8192.0:0.0;
     x->x_outlet1 = outlet_new(&x->x_obj, &s_float);
     if (f == 0) x->x_outlet2 = outlet_new(&x->x_obj, &s_float);
     pd_bind(&x->x_obj.ob_pd, pd_this->pd_bendin_sym);
@@ -328,12 +335,12 @@ static void bendin_list(t_bendin *x, t_symbol *s, int argc, t_atom *argv)
     if (x->x_channel != 0)
     {
         if (channel != x->x_channel) return;
-        outlet_float(x->x_outlet1, value);
+        outlet_float(x->x_outlet1, value + x->x_offs);
     }
     else
     {
         outlet_float(x->x_outlet2, channel);
-        outlet_float(x->x_outlet1, value);
+        outlet_float(x->x_outlet1, value + x->x_offs);
     }
 }
 
@@ -345,7 +352,7 @@ static void bendin_free(t_bendin *x)
 static void bendin_setup(void)
 {
     bendin_class = class_new(gensym("bendin"), (t_newmethod)bendin_new,
-        (t_method)bendin_free, sizeof(t_bendin), CLASS_NOINLET, A_DEFFLOAT, 0);
+        (t_method)bendin_free, sizeof(t_bendin), CLASS_NOINLET, A_GIMME, 0);
     class_addlist(bendin_class, bendin_list);
     class_sethelpsymbol(bendin_class, gensym("midi"));
 }
@@ -355,11 +362,7 @@ void inmidi_pitchbend(int portno, int channel, int value)
     if (pd_this->pd_bendin_sym->s_thing)
     {
         t_atom at[2];
-        // AG: -legacy behavior was changed so that it is consistent with
-        // vanilla bendin.
-        extern int sys_legacy;
-        int shift = sys_legacy ? 0 : 8192;
-        SETFLOAT(at, value-shift); // Ico fix the offset of the incoming pitchbend
+        SETFLOAT(at, value);
         SETFLOAT(at+1, (channel + (portno << 4) + 1));
         pd_list(pd_this->pd_bendin_sym->s_thing, &s_list, 2, at);
     }
-- 
GitLab