



# Make Noise Maths Behavioral Specification for a VCV Rack 2 Emulation

## Executive summary

Dragon King Leviathan, the “engineering takeaways” that matter most for a faithful digital Maths are these:

- The **core outer channels (1 & 4)** are a **dual-mode, direct-coupled slope/function generator**: in **Signal In** mode they behave as a slew/lag processor (and thus naturally implement ASR and envelope following); in **Trigger In** mode they generate a **transient 0→10 V→0 envelope with no sustain**, and are **re-triggerable only during the Falling segment** (not during Rising). 1
- The **EOR/EOC “event” outputs are not documented as narrow pulses**; they are described as **binary 0/10 V “event signals”** with **stable default states** (EOR defaults LOW when idle; EOC defaults HIGH when idle). In cycle use, this strongly implies **gate-like phase indicators** (“high during a segment”), not 1 ms triggers. 2
- **Cycle** in the 2013 revision adds a **Cycle Input**: “Gate HIGH cycles on; Gate LOW cycles off”, with **HIGH  $\geq$  +2.5 V**. This is functionally a **run/stop gate** for self-cycling. 3
- The **BOTH CV input** is explicitly **bipolar exponential** and **inverted in sign** relative to Rise/Fall CV: **positive BOTH speeds up (shorter total time); negative BOTH slows down**. Rise/Fall CV are explicitly **linear**, and **positive increases time** while negative decreases. 4
- The **Vari-Response** knob is explicitly described as a continuous curve family **LOG → LIN → EXPO → HYPER-EXPO** (2013 revision) and defines its “LOG vs EXPO” direction by **slope increasing/decreasing with voltage**. The manual explicitly states that changing response **also changes timing** (“affects Rise and Fall Times”). 5
- **Channels 2 & 3 are attenuverting offset/mix channels**. Their inputs are **normalized to reference voltages** when unpatched to generate DC offsets: **Classic: CH2 and CH3 normalize to +5 V; 2013 revision: CH2 normalizes to +10 V, CH3 to +5 V**. 6
- The **Variable Outs (1–4)** are **normalled into the SUM and OR busses**, and **patching a cable into a Variable Out removes that channel from both busses**. This “break-normal” behavior is essential to emulate for correct bus interaction. 7
- **OR is analog MAX over positive voltages only**: it “outputs the highest voltage” but **does not respond to negative voltages**, so in software it should behave like `max(0, inputs...)` (post-attenuverter). 8
- There is **no publicly available official Maths schematic** in the primary sources retrieved here; therefore, **exact knob laws, CV scalings, comparator hysteresis, and analog saturation points require measurement** if you want a “hardware-indistinguishable” emulation. (Measurement plan provided below.) 9

# Module I/O map and signal conventions

## Versioning used in this report

- “Classic” = original *Maths Classic* (often referred to in community as “Maths 1”). <sup>10</sup>
- “Revision 2013” = *Maths revision 2013* manual (often referred to in community as “Maths 2”). The manual explicitly calls it the “direct descendant” of the original with upgrades and evolutions. <sup>11</sup>
- A community note indicates that the older unit’s physical LFO/cycle switch state may persist across power-off, while the newer does not (relevant only to hardware UX; in Rack you’ll typically serialize state in the patch). <sup>12</sup>

## Global voltage conventions (module-side)

### Verified (Revision 2013):

- Signal inputs (CH1 & CH4) are **direct-coupled** and specified as **±10 V** range. <sup>13</sup>
- Rise CV / Fall CV / Both CV inputs on CH1 & CH4 are specified as **±8 V** range. <sup>13</sup>
- Variable outputs (1–4) are specified as **±10 V** output range. <sup>14</sup>
- SUM output is specified as **±10 V**. INV SUM output is also **±10 V**. <sup>15</sup>
- OR output is specified as **0–10 V** and documented as “does not respond to negative voltages.” <sup>15</sup>
- EOR and EOC outputs are **binary levels: 0 V or 10 V**. <sup>16</sup>

### Verified (Classic):

- Trigger-generated transient is described as a **0 V → 10 V → 0 V** envelope. <sup>17</sup>
- CH2 and CH3 Signal inputs are **normalized to +5 V** for offsets. <sup>18</sup>

### Version conflict to track (input range wording):

- Revision 2013 explicitly says CH1/CH4 Signal In range **±10 V**, but CH2/CH3 Signal In list “Input Range +/-10Vpp.” <sup>19</sup>
- **Inferred (software choice):** treat all **Signal In** jacks as supporting at least **±10 V** in Rack emulation (with soft/hard clipping configurable), because the module is designed as a “direct coupled” processor across channels and the same manual elsewhere uses ±10 V language for outer inputs. <sup>20</sup>

## Normalizations and busses

### Verified (Revision 2013):

- The **Variable Outs 1–4** are **normalised to the SUM and OR busses**. If *no cable* is patched into a Variable Out jack, that channel’s attenuverted signal is injected into the bus; if a cable *is* patched, that channel is removed from the busses. <sup>15</sup>
- **Unity Signal Outs (CH1 & CH4)** are *not* normalised into the SUM/OR bus path, and patching them **does not remove** anything from SUM/OR busses. <sup>21</sup>

### Verified (Classic):

- All four **Signal OUT** jacks are normalized to SUM/OR, and patching removes them from the busses.  
18
- Classic provides a **Signal OUT Multiple** (CH1 & CH4) that is *not* normalized to the busses and therefore does not remove the channel from SUM/OR.  
18

### I/O-by-channel

Below, “panel intent” is what the manual states the control/jack is for; “electrical behavior” is what must be emulated.

#### Channel 1

##### Revision 2013 (Verified):

- **Signal Input:** direct-coupled input to the CH1 circuit; used for lag/portamento/ASR; specified range **±10 V.**  
22
- **Trigger Input:** gate/pulse triggers CH1 regardless of Signal Input activity; result described as **0→10 V transient**, shaped by Rise/Fall/Vari-Response.  
22
- **Cycle Button + Cycle LED:** engages/disengages self-cycle.  
22
- **Rise knob:** CW increases rise time.  
23
- **Fall knob:** CW increases fall time.  
23
- **Rise CV (linear, ±8 V):** positive increases rise time; negative decreases relative to knob setting.  
23
- **Fall CV (linear, ±8 V):** positive increases fall time; negative decreases relative to knob setting.  
23
- **Both CV (bipolar exponential, ±8 V):** positive decreases total time; negative increases total time.  
23
- **Vari-Response knob:** continuously variable LOG→LIN→EXPO→HYPER-EXPO; tick indicates linear.  
24
- **Cycle Input:** gate-controlled cycle state; HIGH requires  $\geq +2.5\text{ V}$ ; low disables cycling unless Cycle button engaged.  
25
- **EOR Output + LED:** “End of Rise Output”; **0/10 V**; defaults low when idle; goes high when the channel reaches its highest voltage; described as useful for clock/pulse; Rise time sets delay-to-high.  
26
- **Unity Signal Output:** tapped from core, not affected by CH1 attenuverter; documented as **0–8 V when cycling**; otherwise follows input amplitude.  
27
- **CH1 Attenuverter (center column):** scales/inverts signal processed/generated by CH1; drives **Variable Out 1** and bus injection.  
28
- **Variable Out 1:** attenuverted CH1 output; **±10 V range**; normalised to SUM/OR unless patched.  
15

##### Classic (Verified differences):

- No Cycle Input jack; “Activity/Cycle Switch” is the combined LED + pushbutton.  
18
- No “Unity Signal Out”; instead there is **Signal OUT Multiple** that duplicates the main output without breaking bus normalization.  
10
- Vari-Response is described as LOG→LIN→EXPO (no explicit Hyper-Expo wording).  
18

## Channel 4

### Revision 2013 (Verified):

- Functional symmetry with CH1 for Signal/Trigger, Rise/Fall controls, Rise/Fall/Both CV ( $\pm 8$  V), Vari-Response, Cycle button/LED, Cycle Input ( $\geq +2.5$  V). <sup>29</sup>
- **EOC Output + LED:** “End of Cycle Output”; **0/10 V**; defaults **HIGH when idle**; event is reaching lowest voltage. <sup>30</sup>
- **Unity Signal Output:** tapped from core; **0-8 V when cycling**, otherwise follows input amplitude. <sup>31</sup>
- **CH4 Attenuverter** drives **Variable Out 4** and bus injection. <sup>32</sup>

### Classic (Verified differences):

- Same “Activity/Cycle Switch” concept as CH1; no Cycle Input jack. <sup>17</sup>
- Has **Signal OUT Multiple**. <sup>18</sup>

## Channels 2 and 3

### Revision 2013 (Verified):

- **CH2 Signal Input:** direct-coupled; normalized to **+10 V** reference for offset generation (when unpatched). <sup>33</sup>
- **CH3 Signal Input:** direct-coupled; normalized to **+5 V** reference for offset generation. <sup>33</sup>
- **CH2 & CH3 Attenuverters:** provide scaling/attenuation/inversion; manual explicitly also says “amplification” for these channels. <sup>33</sup>
- **Variable Out 2 / 3:** attenuverted signals;  **$\pm 10$  V range**; normalled into SUM/OR unless patched. <sup>15</sup>

### Classic (Verified):

- CH2 and CH3 inputs normalized to **+5 V** reference. <sup>18</sup>
- Scaling/Inversion control is explicitly described: **NOON = zero output**, full CW = maximum positive, full CCW = maximum inverted; CH2 and CH3 offer “a small amount of gain.” <sup>18</sup>

## Bus and global outputs

### Revision 2013 (Verified):

- **OR OUT:** analog OR; “always outputs the highest voltage” among CH1-CH4 variable outs; does not respond to negative voltages; **0-10 V range**. <sup>34</sup>
- **SUM OUT:** algebraic sum of CH1-CH4 variable outs;  **$\pm 10$  V range**. <sup>34</sup>
- **INV OUT:** inverted SUM;  **$\pm 10$  V range**. <sup>34</sup>

### Classic (Verified):

- SUM and OR bus exist, but **no INV SUM output**. <sup>10</sup>

# Channel 1 & 4 function generator model

This section is split into **Verified** behavior (explicit in manuals) and **Inferred** DSP-ready modeling choices.

## Verified behavioral rules

### Dual role: direct-coupled processing + function generation

- Signal inputs are **direct coupled** (audio + CV capable) and used for lag/portamento/ASR processing.  
35
- With **Trigger Input**, the channel produces a transient function described as: rises **0→10 V**, then immediately falls **10→0**, with **NO SUSTAIN**.  
36

### Attack/decay and retriggers

- **Rise** determines the time to travel upward to the maximum voltage; CW increases time.  
4
- **Fall** determines the time to travel downward to minimum voltage; CW increases time.  
4
- Triggered transient is **re-triggerable during the Falling portion but not during the Rising portion**, enabling clock/gate division by making Rise long enough to ignore incoming triggers.  
37
- Trigger input is explicitly called useful for "LFO Reset (only during Falling portion)." 38

### Rise/Fall CV vs Both CV

- **Rise CV** and **Fall CV** are explicitly "linear" control inputs; **positive increases time** (slower), negative decreases time (faster), relative to knob setting.  
4
- **Both CV** is explicitly "bipolar exponential"; **positive decreases total time** (faster), negative increases total time (slower).  
4
- The manual explicitly reiterates: Both CV changes the rate of entire function and is **inverse** in polarity to Rise/Fall CV inputs.  
39

### Curve family

- Vari-Response shapes Rise/Fall behavior continuously; manual defines:
- LOG: rate of change decreases as voltage increases
- EXPO: rate of change increases as voltage increases
- LINEAR: constant rate  
and includes "everything in-between."  
40
- The manual explicitly states response curve adjustment **affects Rise and Fall Times**.  
41

### Cycle behavior

- In revision 2013, Cycle can be engaged by **Cycle button** or **Cycle Input**; both "do the same thing" (self-oscillate).  
42
- Cycle Input is explicitly gate-qualified, with **HIGH ≥ +2.5 V**.  
43
- Patch example explicitly uses **Cycle Input for Run/Stop control** of pulse/clock behavior.  
44

## EOR/EOC semantics and default states

- **EOR (CH1):** binary 0/10 V; defaults low when idle; “event” is reaching highest voltage. <sup>26</sup>
- **EOC (CH4):** binary 0/10 V; defaults high when idle; “event” is reaching lowest voltage. <sup>30</sup>
- The closely-related Make Noise <sup>45</sup> module FUNCTION <sup>46</sup> (same “core circuit” family) specifies an **EOR/EOC LED state model consistent with complementary gate states:** “When Red, EOC is High and EOR is Low; when Green, EOR is High and EOC is Low.” <sup>47</sup>

## Time range

- Maths is described as capable of times **as slow as ~25 minutes** and **as fast as ~1 kHz** (audio rate), with mention of adding external control signals for “slow-ver-drive.” <sup>48</sup>

## Inferred DSP model that matches verified constraints

Because no official schematic-level transfer functions were available in the accessible primary sources here, the following is an **inferred** model designed to be *implementation-ready* while matching all documented behaviors.

### State machine

Model each outer channel as a **stateful slope generator** with three primary states:

- **IDLE** (resting at low endpoint unless tracking Signal In)
- **RISING**
- **FALLING**

and a boolean latch `cycleEnabled`.

**Verified constraints the model must satisfy:** - Trigger causes a transient 0→10→0 with no sustain. <sup>35</sup>

- Retrigger is ignored during RISING but accepted during FALLING. <sup>49</sup>
- Cycle input/button engages self-oscillation; Cycle Input is gate-controlled with threshold +2.5 V. <sup>25</sup>
- EOR/EOC are 0/10 “event” signals with the described defaults. <sup>50</sup>

### Inferred pseudocode (DSP-ready)

(Use this as a baseline, then refine constants by measurement.)

```
constants:  
V_LOW = 0.0  
V_HIGH = 10.0           // manual's triggered transient peak  
CYCLE_HIGH_THRESH = 2.5 // V, per manual  
  
state per voice:  
float y                // internal slope output (pre-attenuverter)  
enum Phase {IDLE, RISING, FALLING}  
bool transientMode     // true if Trigger In initiated the segment pair  
float riseTimeSec, fallTimeSec // effective after CV + response coupling
```

```

    float shapeParam      // normalized vari-response (-1 log .. 0 lin .. +1
hyper-exp)
    bool cycleLatchedButton // UI param
    bool cycleGateHigh     // from Cycle input >= thresh
    bool trigEdge          // rising edge detected at Trigger input
    float xSignalIn        // signal input (direct-coupled)

inputs->mode selection (inferred):
cycleEnabled = cycleLatchedButton OR (cycleGateHigh)

Trigger handling:
if trigEdge:
    if Phase != RISING:           // verified: ignore during RISING
        transientMode = true
        if Phase == IDLE: y = V_LOW           // start from 0 when idle
        // if Phase == FALLING, keep current y to avoid discontinuity (inferred)
        Phase = RISING

Cycle start:
if Phase == IDLE and cycleEnabled:
    transientMode = true
    y = V_LOW
    Phase = RISING

Segment targets:
if transientMode:
    targetRise = V_HIGH
    targetFall = V_LOW
else:
    // slew/track mode:
    targetRise = xSignalIn
    targetFall = xSignalIn

Per-sample update:
if Phase == RISING:
    y += stepToward(targetRise, riseTimeSec, shapeParam)
    if y >= targetRise - eps:
        y = targetRise
        Phase = FALLING
    else if Phase == FALLING:
        y += stepToward(targetFall, fallTimeSec, shapeParam)
        if y <= targetFall + eps:
            y = targetFall
            if transientMode and cycleEnabled:
                Phase = RISING      // self-cycle
            else:
                Phase = IDLE
                transientMode = false

```

```

EOR/EOC outputs (inferred from manuals + FUNCTION behavior):
  EOR = (Phase == FALLING) ? 10V : 0V           // matches EOR default low, goes
high at end of rise
  EOC = (Phase != FALLING) ? 10V : 0V           // matches EOC default high, goes
low during fall

```

### Shape function `stepToward()` and the Vari-Response “time coupling”

**Verified requirements:** - LOG means slope decreases as voltage increases; EXPO means slope increases as voltage increases. <sup>51</sup>

- Changing response curve affects Rise/Fall times. <sup>52</sup>

**Inferred implementation approach:** 1. Compute a **base time** from Rise/Fall knobs + CV.  
 2. Apply a **shape-dependent time scale** (because the manual says shape affects time).  
 3. Apply a **shape-dependent curvature** within the segment.

A practical, stable formulation:

- Let `u` be normalized progress through the segment (0 at start, 1 at target).
- Use `curve(u, p)` where:
  - `p < 1` produces fast-start/slow-end (manual's LOG direction for rising)
  - `p = 1` linear
  - `p > 1` slow-start/fast-end (manual's EXPO direction)

Then: - `p = map(shapeParam)` where `shapeParam` maps the knob (LOG..LIN..EXPO..HYPER-EXPO) into a power/exponent range.

**Versioning note (Verified + Inferred):** - Revision 2013 explicitly adds “greater logarithmic range” and extends response to “Hyper-Exponential.” <sup>53</sup>

- **Inferred consequence:** v2 should allow a wider exponent range (especially on the LOG side for “portamento” feel) than Classic.

### Rise/Fall vs Both CV interaction

**Verified facts:** - Rise/Fall CV are linear; Both CV is exponential; Both polarity speeds up on positive. <sup>4</sup>

**Inferred combined timing model (recommended for emulation):** - Treat `Rise` and `Fall` knobs as defining a base time constant (logarithmic knob law likely; see measurement plan).  
 - Treat RiseCV and FallCV as **linear offsets in a “time control” domain** (not necessarily linear in seconds).  
 - Treat BothCV as **multiplicative scaling of the rate** (exponential in time), i.e. `time *= exp(-k * bothCV)` with `k` to be measured.

## Slew limiter mode behavior (Signal Input)

- Verified:** - MATHS cannot “increase the rate” of an external voltage; it can only slow it down or allow it through. <sup>54</sup>
- A gate applied to Signal Input gives an ASR envelope: rise to gate level, sustain until gate ends, then fall to 0. <sup>55</sup>

- Inferred (implementation):** - In Signal Input mode, treat the channel as an **asymmetric slew limiter**: it tracks the input with a limited  $dV/dt$  on rises (Rise time) and on falls (Fall time), and applies the same curvature family as in triggered mode (because the manual says Rise/Fall/Vari-Response shape responses to signals applied to Signal Input). <sup>54</sup>
- Ensure stability for audio-rate Signal In: for high-frequency inputs with fast Rise/Fall, you may need **oversampling** if you nonlinearly shape or hard-clip.

## Channels 2 & 3 attenuverting / mixing

### Attenuverter behavior and offsets

- Verified (Classic):** - Channels 2 and 3 generate offsets when unpatched because their inputs are normalized to +5 V. <sup>18</sup>
- The Scaling/Inversion control is explicitly defined: **NOON = zero**, full CW maximum, full CCW maximum inverted; CH2/CH3 have “a small amount of gain.” <sup>18</sup>

- Verified (Revision 2013):** - CH2 input is normalized to **+10 V** reference; CH3 to **+5 V** reference (offset generation). <sup>56</sup>
- CH2/CH3 attenuverters provide scaling/attenuation/inversion and are described as also providing “amplification.” <sup>33</sup>

- Inferred (implementation details):** - Implement CH2/CH3 attenuverters as a bipolar gain control around 0 at noon, with at least `gain ∈ [-1, +1]`.
- Provide an optional “gain > 1” headroom (e.g. up to 1.2–2.0) *only if measurement confirms it materially affects feel*; Classic explicitly hints at “small amount of gain” on CH2/3. <sup>18</sup>
- Offset when unpatched: - v1 Classic: `x2_norm = +5 V`, `x3_norm = +5 V`. <sup>18</sup>
- v2 2013: `x2_norm = +10 V`, `x3_norm = +5 V`. <sup>33</sup>

### SUM and OR mixing behavior

- Verified:** - **SUM** is the analog sum of the four attenuverted (variable) channel outputs. <sup>34</sup>
- **OR** “always outputs the highest voltage” among its inputs, and “does not respond to negative voltages.” <sup>57</sup>
- Practical patch notes reinforce OR’s rectifying behavior: half-wave rectification uses OR output. <sup>58</sup>

- Inferred (DSP mapping):** - Let `v1..v4` be Variable Out signals *only if* their corresponding output jack is **unpatched** (break-normal). Otherwise, that channel contributes **0** to busses. <sup>57</sup>
- **OR output:** `or = max(0, v1, v2, v3, v4); clamp to [0, 10].` <sup>34</sup>

- **SUM output:** `sum = v1 + v2 + v3 + v4`; clamp/saturate around  $\pm 10$  V. 15

- **INV output** (v2 only): `inv = -sum`. 34

## Diode-drop / non-idealities

**Verified:** The manuals describe idealized behavior (max selector, sum, no negative response) without specifying diode drops or precision rectification. 57

### Inferred:

- Real analog OR circuits may exhibit diode drops unless compensated; whether Maths does is unknown without schematic/measurement. This is likely perceptually minor for control voltages but might affect precise rectification.
- Recommendation: emulate as **ideal max/rectifier** by default, and optionally include a tiny "drop" mode (e.g. 0.1-0.3 V) toggled via a "hardware non-ideality" setting only if measurement demonstrates it. (See measurement plan.)

## Comparator/logic specifics

### Gate/trigger thresholds

**Verified (Cycle Input threshold):** - Cycle Input requires **minimum +2.5 V** for HIGH. 43

**Verified (Logic outputs are 0/10 V):** - EOR/EOC are explicitly 0/10 V. 59

**Inferred (Trigger input threshold and hysteresis):** - The manuals do **not** specify Trigger Input threshold, hysteresis, or minimum pulse width. 1

- A safe Rack implementation can adopt a Schmitt trigger detector (see Rack guidelines below), but if you aim for hardware parity you must measure actual thresholds.

### EOR/EOC "pulse width" interpretation

**Verified textual evidence:** - EOR/EOC are described as "event signals" that go high at the end of rise / end of fall and have defined idle defaults. 50

- Patch guides treat EOR/EOC as **pulse/clock sources with variable width**, and explicitly say: - Using EOR: Rise more effectively adjusts frequency, Fall adjusts pulse width
- Using EOC: Rise more effectively adjusts width, Fall adjusts frequency 60
- The related FUNCTION module states EOR/EOC LED indicates mutually exclusive high states, which is consistent with phase gates rather than narrow pulses. 47

**Inferred conclusion (strong):** - Emulate EOR/EOC as **gate outputs indicating phase** (EOR high during FALLING, EOC high during RISING/IDLE), not short trigger pulses—unless measurement contradicts.

### "Comparator / gate extractor" edge cases

**Verified (Patch behavior indicating internal timing logic):** - The manual's "Comparator/Gate Extractor (A New Take)" patch indicates: - With CH1 Rise/Fall at 0 (fast), **EOR trips when the signal goes slightly**

**positive.**

- CH1 Fall lengthens the derived gates; CH1 Rise sets the required time above threshold to trip. 61

**Inferred implication:** - This behavior is consistent with a slope-based comparator: a threshold crossing causes the slope generator to switch phase and thus toggle EOR gating, effectively turning analog threshold crossings into gates with controllable delay/width.

## Calibration, tolerances, and “feel”

### What is known from primary sources

**Verified:** - Revision 2013 lists several circuit/layout evolutions, including:

- Signal Output Multiple replaced by Unity Signal Output
- Added INV SUM output
- Added LEDs for SUM and EOR/EOC state
- EOC output buffered for stability
- Added Cycle Input (gate-controlled cycling)
- Added offset ranges ( $\pm 10$  on CH2 or  $\pm 5$  on CH3)
- Greater LOG range in Vari-Response for “East Coast style Portamento” 53

### What is not specified but is likely “feel-critical”

**Inferred (requires measurement to perfect):** - **Rise/Fall knob law:** almost certainly nonlinear (huge time span), likely closer to exponential/log taper than linear seconds-per-degree. 51

- **Both CV scaling:** specified as exponential but not quantified (e.g., V/oct or V/decade). 23
- **Response curve mapping:** the exact transfer from knob position to curve exponent and the “time coupling” factor is undocumented. 52
- **Saturation/clipping points:** outputs are specified as ranges ( $\pm 10$ , 0–10), but actual analog headroom/soft clipping shapes are undocumented. 15
- Patch “Pseudo-VCA with clipping” explicitly relies on audible clipping behavior when adding offsets and mixing; that suggests non-ideal saturation matters to some patches. 61

### Emulate vs ignore: recommendations

**Recommended to emulate (high perceptual impact):** - Retrigger-only-on-fall rule. 49

- Cycle Input gate threshold (+2.5 V) and run/stop behavior. 62
- OR = max over positives only with 0 baseline. 34
- Vari-Response curve family and the fact that it changes timing. 52

**Reasonable to approximate unless doing “gold standard” emulation:** - Exact diode drop in OR.

- Precise analog saturation curvature at  $\pm 10$  V.
- Exact hysteresis thresholds of trigger comparators (except Cycle Input which is specified). 43

# Experimental measurement plan

If you want the emulation to feel indistinguishable from hardware, measure these items on at least one unit per revision (Classic + 2013).

## Gear and capture requirements

- **DC-coupled audio interface** (or oscilloscope with DC coupling).
- Sample at  $\geq 96$  kHz for accurate edge timing and audio-rate cycling behavior; higher (192 kHz) if you want tight 1 kHz+ measurements.
- Record at **24-bit** to keep quantization noise out of very slow slopes.
- For CV sweeps, a calibrated CV source (or a precision DAC module) is ideal.

## Test patches and what to record

### Function generator transient (Trigger In mode)

- Patch: nothing to Signal In; feed a single trigger into Trigger In; monitor Unity Out and Variable Out.
- Measure: - Peak amplitude at each output (is Unity 8 V or 10 V in this mode?). [25](#)
- Rise time and fall time vs knob positions (multiple points). [39](#)
- Retrigger behavior: fire a second trigger at multiple phases (early rise, late rise, early fall, late fall). Confirm ignore vs restart semantics. [63](#)

### Cycle mode amplitude and frequency mapping

- Patch: engage cycle; monitor Unity Out, Variable Out, EOR/EOC.
- Measure: - Unity Out amplitude (should be 0–8 V when cycling per manual) and whether Variable Out differs. [43](#)
- Frequency vs Rise/Fall settings at multiple curve settings. [64](#)
- Effect of Vari-Response on total period (quantify “time coupling”). [52](#)

### Cycle Input run/stop edge cases (v2)

- Patch: keep Cycle button off; apply gate to Cycle Input; then remove gate mid-cycle.
- Measure: - Whether oscillation stops immediately or at end-of-cycle (phase). [62](#)
- Minimum voltage for reliable HIGH detection (should be ~+2.5 V). [65](#)

### EOR/EOC width and phase

- Patch: cycle at slow rate; monitor EOR/EOC with scope.
- Measure: - Whether EOR/EOC are “phase gates” (high for the duration of a segment) or narrow pulses. [66](#)
- Decide which internal state generates them (RISING vs FALLING) and confirm idle defaults. [50](#)

### OR and SUM non-idealities

- Patch: feed known DC levels into CH2/CH3 and known LFOs into CH1/CH4; compare OR output to a software max.
- Measure: - Any voltage drop from OR relative to max input. [34](#)
- SUM saturation onset and whether clipping is hard or soft. [67](#)

## Table template for results

Use a table per revision and per output point.

| Test              | Knob/CV settings                   | Output monitored | Metric     | Measured value | Notes |
|-------------------|------------------------------------|------------------|------------|----------------|-------|
| Trigger transient | Rise=0.25, Fall=0.25,<br>Shape=LIN | Unity Out        | Peak (V)   |                |       |
| Trigger transient | same                               | Variable Out     | Peak (V)   |                |       |
| Cycle             | Rise=0.5, Fall=0.5,<br>Shape=LOG   | Unity Out        | Period (s) |                |       |
| Cycle             | same                               | EOR              | Duty %     |                |       |
| OR                | v1=3V, v2=-2V, v3=5V,<br>v4=4V     | OR Out           | Output (V) |                |       |
| SUM               | same                               | SUM Out          | Output (V) |                |       |

For curve fitting: - Fit Rise/Fall knob to time using a **log mapping** first (e.g.  $T = T_{min} * (T_{max}/T_{min})^k$ ) and refine by least squares. - Fit Both CV scaling as an **exponential in time** ( $T \propto \exp(-a*V)$ ) unless measurements show otherwise (e.g., piecewise).

## Implementation notes for VCV Rack 2

### Rack voltage/trigger conventions (software-side)

VCV's own manual recommends:

- **Gates:** 10 V when active.
- **Triggers:** 10 V, duration about 1 ms; detect using Schmitt trigger with low threshold ~0.1 V and high threshold ~1–2 V (example code uses `dsp::SchmittTrigger`). <sup>68</sup>
- **Polyphony:** cables can carry up to 16 channels; modules should process each channel independently. <sup>69</sup>

These standards are about Rack ecosystem compatibility and are not necessarily identical to Maths hardware thresholds—so if you emulate hardware thresholds, still consider providing Rack-friendly behavior for user experience.

### DSP architecture recommendation

Implement Maths as **four per-voice channel blocks + shared bus block**:

- **CH1 block (polyphonic):**

- Slope generator state machine + CV parameter evaluation (Rise/Fall/Both/Shape)
- Outputs: UnityOut1, VarOut1, EOR
- **CH4 block (polyphonic):**
- Same + EOC
- **CH2/CH3 blocks (polyphonic):**
- Input normalization (offset) + attenuverter gain
- Outputs: VarOut2/VarOut3
- **Bus block (polyphonic):**
- For each poly channel index  $c$ , compute  $\text{busIn}[i]$  as  $\text{VarOut}[i]$  only if that output jack is **unpatched** (normalled). 57
- Compute OR/SUM/INV.

## Parameter smoothing / time discretization

- Rack's engine already includes smoothing behavior for parameters in some contexts, but you should not rely on it for "analog feel" because envelope timing is extremely sensitive. 70
- Recommended:
- Smooth raw knob/CV-derived **time constants** (Rise/Fall/Both scaling) with a short one-pole (e.g., 1–5 ms) to avoid zipper noise at audio-rate cycling.
- Do **not** smooth Trigger edges; detect edges with Schmitt trigger and feed a one-sample event into your state machine.

## Oversampling

- If your implementation includes **nonlinear saturation** (soft clipping) or a **nonlinear curve that depends on signal amplitude**, audio-rate cycling can alias.
- Recommendation: offer optional **2x oversampling** for CH1/CH4 signal generation paths only (not needed for slow CV use), controlled by a "HQ" switch.

## Polyphony strategy

- Treat polyphonic cables as **independent copies of the entire Maths circuit per channel index** (including each channel's internal state). This matches Rack's polyphony model expectations. 69
- For performance, it's acceptable to keep CH2/CH3 purely per-channel scalar math, but CH1/CH4 require per-channel state arrays.

## Deterministic unit tests

Where the manuals specify exact behavior, implement unit tests with known expected outcomes.

### Verified test vectors (examples):

- **Cycle gate threshold (v2):**
- Input Cycle = +2.4 V → cycle should remain off
- Input Cycle = +2.5 V → cycle should be on 43
- **Retrigger rule:**

- When in RISING, Trigger edges must not restart (no phase reset). [63](#)
- When in FALLING, Trigger edge must retrigger (restart rising phase) (exact “reset to 0 vs restart from current level” requires measurement). [63](#)

- **OR behavior:**

- Inputs: [-2, 3, 5, 4] → OR = 5
- Inputs all negative → OR = 0 [34](#)

- **Bus normalization:**

- If VarOut2 jack is patched, CH2 contribution must be removed from SUM and OR busses. [57](#)

**Measurement-dependent (“gold standard”) tests:** - Exact Rise/Fall knob→time mapping curve. - Exact Both CV scale (volts-per-octave or similar). - Exact EOR/EOC “gate vs pulse” timing and duty cycle.

## Known Unknowns

These are the remaining blockers to a hardware-indistinguishable implementation:

- Exact **Rise/Fall knob law** (seconds vs knob position) and whether it is the same in Classic vs 2013 revision. [71](#)
- Exact **Both CV exponential scaling** (units, slope) and how it composes with Rise/Fall CV. [23](#)
- Whether Trigger retrigger during FALL **resets to 0 or restarts from current voltage** (manual only specifies phase allowance, not restart semantics). [63](#)
- Whether Unity outputs are **0–8 V only in cycle** or also in triggered mode, and how that relates to the manual’s “0–10 V function” language. [72](#)
- Whether OR output exhibits any diode drop / compression, and where SUM clips (hard vs soft). [73](#)
- Trigger input threshold, hysteresis, and minimum trigger width (undocumented in manuals). [74](#)
- Any hardware calibration/trimmers affecting timing (not mentioned in accessible primary sources). [75](#)

## Bibliography

Primary sources (Make Noise):

- Maths revision 2013 manual (PDF). [76](#)
- Maths Classic manual (PDF). [17](#)
- FUNCTION manual (PDF), for corroborating EOR/EOC semantics and “same core circuit family” behavior. [77](#)
- Maths product page (time range statement is repeated there). [78](#)

VCV Rack developer references:

- VCV Rack Manual: Voltage Standards (triggers/gates thresholds and suggested Schmitt trigger usage). [68](#)

- VCV Rack Manual: Polyphony. <sup>79</sup>
- VCV Rack Manual: Plugin API Guide (polyphony implementation guidance). <sup>80</sup>

Secondary/community reference used sparingly (version UX note):

- Sequencer.de forum thread noting hardware cycle switch persistence difference between older and newer Maths (non-primary, UX-only). <sup>12</sup>

---

[1](#) [2](#) [3](#) [4](#) [5](#) [7](#) [8](#) [9](#) [11](#) [13](#) [14](#) [15](#) [16](#) [19](#) [20](#) [21](#) [22](#) [23](#) [24](#) [25](#) [26](#) [27](#) [28](#) [29](#) [30](#) [31](#) [32](#) [33](#) [34](#) [35](#)  
[36](#) [37](#) [38](#) [39](#) [40](#) [41](#) [42](#) [43](#) [44](#) [45](#) [46](#) [48](#) [49](#) [50](#) [51](#) [52](#) [53](#) [54](#) [56](#) [57](#) [58](#) [59](#) [60](#) [61](#) [62](#) [63](#) [64](#) [65](#) [66](#) [67](#)

[71](#) [72](#) [73](#) [74](#) [75](#) [76](#) <https://www.makenoisemusic.com/wp-content/uploads/2024/03/MATHSmanual2013.pdf>

<https://www.makenoisemusic.com/wp-content/uploads/2024/03/MATHSmanual2013.pdf>

[6](#) [10](#) [17](#) [18](#) [55](#) <https://www.makenoisemusic.com/wp-content/uploads/2024/03/MATHSCLASSICmanual.pdf>

<https://www.makenoisemusic.com/wp-content/uploads/2024/03/MATHSCLASSICmanual.pdf>

[12](#) <https://www.sequencer.de/synthesizer/threads/make-noise-maths.74948/>  
<https://www.sequencer.de/synthesizer/threads/make-noise-maths.74948/>

[47](#) [77](#) <https://www.makenoisemusic.com/wp-content/uploads/2024/03/FUNCTIONmanual.pdf>  
<https://www.makenoisemusic.com/wp-content/uploads/2024/03/FUNCTIONmanual.pdf>

[68](#) <https://vcvrack.com/manual/VoltageStandards>  
<https://vcvrack.com/manual/VoltageStandards>

[69](#) [79](#) <https://vcvrack.com/manual/Polyphony>  
<https://vcvrack.com/manual/Polyphony>

[70](#) <https://community.vcvrack.com/t/parameterquality-change-in-2-3-0/19533>  
<https://community.vcvrack.com/t/parameterquality-change-in-2-3-0/19533>

[78](#) **MATHS – Make Noise Music**  
<https://www.makenoisemusic.com/modules/math/>

[80](#) <https://vcvrack.com/manual/PluginGuide>  
<https://vcvrack.com/manual/PluginGuide>