From 166008f64f356fc9819c174f9a978443b1237e25 Mon Sep 17 00:00:00 2001 From: Gordon JC Pearce Date: Wed, 8 Jan 2025 13:48:57 +0000 Subject: [PATCH] text files --- README.md | 37 ++++++++++++++++++++++++++++++++ TECHNICAL.md | 58 +++++++++++++++++++++++++++++++++++++++++++++++++++ alphaOsc.jpg | Bin 0 -> 9692 bytes 3 files changed, 95 insertions(+) create mode 100644 README.md create mode 100644 TECHNICAL.md create mode 100644 alphaOsc.jpg diff --git a/README.md b/README.md new file mode 100644 index 0000000..5d1b64a --- /dev/null +++ b/README.md @@ -0,0 +1,37 @@ +alphaOsc +======== +alphaOsc is a proof-of-concept of a Roland Alpha Juno DCO written in C++. This all stems from a [discussion thread](https://www.kvraudio.com/forum/viewtopic.php?t=420194) on the KVRAudio forums. + +![alphaOsc screenshot][alphaOsc.jpg] + +Description +----------- +This is just the bare bones oscillator, based around what I'm guessing is in the Alpha Juno's 87123 DCO chip. It does not work particularly well and it is not particularly efficient. It does not have any anti-aliasing and does not perform well much above middle C. + +WARNING! alphaOsc produces a lot of sharp unfiltered squarewaves with loads of high-frequency content. If you are exposed to alphaOsc's harsh and unsettling racket, you only have yourself blame. + +Building +-------- +alphaOsc is written using [Distrho Plugin Framework](https://github.com/DISTRHO/DPF). Once you have cloned the git repository, pull down the DPF submodule: + + git submodule update --init --recursive + +and build alphaOsc: + + make + +This will compile the jack standalone version and LV2 version. It ought to be possible to also compile VST/VST3 and AU versions but these have not yet been tested. + +Installing +---------- +There are no specific installation instructions because the jack standalone version can just be run as `./bin/alphaOsc`. You can install the LV2 version into your local LV2 plugins with: + + cp -r ./bin/alphaOsc.lv2 ~/.lv2/ + +Using alphaOsc +-------------- +Start alphaOsc either using the jack standalone version or in a plugin host of your choice. Connect a MIDI source to its MIDI input and connect its audio output to something that will let you hear its output. This plugin is monophonic, and sounds the highest note played with a short "queue" of notes for rollover. + +Acknowledgements +---------------- +Massive thanks to [falkTX](https://github.com/falkTX) for writing DPF and all the many amazing folk who contributed to it, itoa, the prolific aciddose, TechnoManiac, and mystran (and probably others) from the KVR Audio forums for nerd-sniping me so hard into doing this, Adam Inglis, Russell McClellan, Tom Wiltshire, and Richie Burnett from the synth-diy mailing list for various helpful discussions over the years about the Alpha Juno, and of course all the unknown heros who scanned and uploaded all those service manuals and circuit diagrams. \ No newline at end of file diff --git a/TECHNICAL.md b/TECHNICAL.md new file mode 100644 index 0000000..5c23586 --- /dev/null +++ b/TECHNICAL.md @@ -0,0 +1,58 @@ +alphaOsc +======== +alphaOsc is a proof-of-concept of a Roland Alpha Juno DCO written in C++. This all stems from a [discussion thread](https://www.kvraudio.com/forum/viewtopic.php?t=420194) on the KVRAudio forums. This discussion was started over ten years ago, got quite heated in places, went quiet for a while, then had a few posts now and again before bursting back into life early in 2024. + +It's easy to chat back and forth about what you think is going on, but it's better to show your working. You know, like your old maths teacher used to tell you? So even if you're wrong people can tell if you're "wrong for the right reasons". This project is my proof-of-concept of my potentially incorrect ideas about how the Alpha Juno works. I won't go into a big long discussion of the Alpha Juno or the "Belgian Hoover" patch because if you have read this far you already know all that. I will talk a little about the hardware of the Alpha Juno first, though, so we can clear up any misunderstandings. + +In the beginning... +------------------- +... there was the Juno 6. This had "DCOs", Digitally Controlled Oscillators, which were really just saw-core VCOs with their integrator reset by a digital clock instead of a comparator. They still needed a pitch control voltage to set the ramp current, but they remained locked in tune with the clock - much like hard sync but without the comparator letting it free-run. This design persisted right through to the Juno 106, and you see the same idea used in many other analogue synths, not just by Roland. + +Going Digital +------------- +For the Alpha Juno Roland took a very different approach, using a custom DCO chip driving six IR3R05 VCF and VCA chips. Sadly there is almost no documentation on Roland's custom ICs and they are no longer available, which makes repairs difficult and experimentation unwise. Unlike the "analogue" Juno family which contain no "black box" custom chips (the IR3109 and BA662s used in the filters are well-known now and the Juno 106 "waveshapers" just have a couple of opamps and a 4013 flipflop in), the Alpha Juno crams a lot of its functionality into three custom chips - IC26, the 63H149 keyboard scanner gate array, IC18, the address decoding and I/O gate array and the one we're interested in here, IC5, the 87123 DCO chip. + +There appears to be absolutely no technical information online about the 87123, so everything presented here is a guess, albeit guesses backed by a bit of evidence. + +The only details in the Alpha Juno Service Manual (it doesn't matter if you pick the manual for the AJ1, AJ2, HS10, or HS80 - this bit is the same) are that it's a 24-pin chip with six waveform outputs, two test pins, a clock pin, eight data pins, an "Address Latch Enable" (ALE) pin, a chip select (/CS, the slash representing the overbar meaning the pin is active low) pin, and analogue and digital +5V supplies. What does this tell us about the chip? + +Wild-eyed Guesswork Begins +-------------------------- +There are eight data pins which line up neatly with the 8-bit CPU IC19, 8032. This part has an external ROM which could be disassembled to find out more information about the 87123, but that's a job for another day. The interesting thing here though is that the 8032 family use the same 8-bit bus on Port 0 for data and the lower 8 bits of the address, and use the ALE pin to latch those address bits into an external 8-bit latch to drive external memory. In this case the latch is in IC18, with the upper ROM and RAM bits coming from IC5's Port 2 pins. + +This suggests that the 87123 has at most 256 8-bit registers to control it. It's probably 16 registers, and here's why I think that: + +The 8253 counters in the Juno 6 family are 16-bit but they have an additional pair of flip-flops in the clock raising them to effectively 18-bit. The counters are loaded with a 16-bit value which counts down to zero, reloading the counter and triggering an output pulse. It seems reasonable to guess that the 87123 uses a similar counter scheme and drives another 8-bit counter with that pulse to generate the sawtooth wave, with counter bits being combined to form the square and "complex" waveforms. Having six 16-bit counters would account for 12 registers. + +The Systems Exclusive information in the MIDI manual has the "switch values" for the DCO, and from that we can see that the pulse waveform, DCO range, sub level, and noise level are each 2-bit values. That's 13 registers. The DCO sub and saw waveforms are 3-bit values, which would be a 14th register. It's possible that the pulse, saw, and sub values are all in one register because that still adds up to 8. + +This leaves the PW Depth and PW Rate registers. I've absolutely no evidence to suggest that the PW LFO is generated within the 87123, it just neatly fits into 16 registers if it does. + +How the Waveforms Might Be Generated +------------------------------------ +The DCO range is set to 4', 8', 16', or 32' by a prescaler consisting of some flip-flops that divides the incoming 12MHz clock down. This is fed to a set of six 16-bit programmable counters, which count down, triggering a counter for each voice. A 10-bit counter would generate the right "divisions" for the two sub oscillator outputs, and the sawtooth output appears to be 8-bit. + +Starting with the pulse wave output, the square wave is just bit 7 of the saw counter fed to the output. It is probably switched on and off by ANDing it with its control pin. The 25% pulse wave is most likely generated by ANDing bit 7 and bit 6 of the saw counter, with the output going high only in the last quarter of the count when both bits are high. + +The PWM gating is probably done by comparing the saw counter with the pulsewidth value, and having it go high when the count is equal to or greater than the pulsewidth setting. Since the PW value is from 0 to 127 this is probably compared against bits 0 to 6 of the saw counter, giving two "pulses" per squarewave. + +When the PWM gate is ANDEd with bit 7 then like the 25% cycle it will be off while bit 7 is low, and then only on for the second half of the squarewave cycle while the PWM gate is high. + +The saw counter's output would quite likely be latched onto a simple R/2R ladder DAC. There doesn't appear to be any capacitors or any signs of multiplexing the outputs, so it may well just have a latch and a DAC for every output. The "straight" sawtooth is turned on and off by enabling the DAC latch, and the "complex" waveforms are generated by ANDing the DAC latch control with other signals. + +To generate the first "50% saw" wave the DAC latch is ANDed with bit 6 of the saw counter, just as for the 25% pulse width setting. For the "comb saw", it is ANDed with bit 4 of the counter - three octaves up - and for the last one it is anded with both bit 6 and bit 4. For the PWM saw wave it is just ANDed with the PWM gate signal, but since this switches twice per cycle you get the 50% saw shape. + +Finally the sub oscillator is generated similarly to the pulse oscillator, but with the outputs being taken from bit 8 or bit 9 of the saw counter and the "chopper" signals taken from bit 7, bit 6, bit 5, or bit 4 accordingly. + +The noise and sub outputs have four levels, which is again probably implemented by something like a resistor ladder and some AND gates. + +Further Work For Someone Else +----------------------------- +To turn this into a full Alpha Juno plugin you would need to determine how the envelope and LFO works along with all the modulation depth controls, possibly by disassembling the Alpha Juno ROM or by studying a real Alpha Juno and carefully measuring its outputs. You would need to implement antialiasing to clean up the waveform. Modelling the IR3R05 filter IC might not be entirely necessary, but the service manual describes it as being a pair of State Variable Filters with lowpass and bandpass outputs exposed. The highpass filter appears similar to the Juno 106, with the lowest setting being a bass boost, the next one up being flat, and then the top two being 1-pole high cut filters at about 220Hz and 700Hz. The chorus circuit is similar to the analogue Junos, with two delay lines driven by an LFO, with the modulation inverted for one of the channels. + +Conclusion +---------- +The proof-of-concept presented here is not a fully-functional plugin but could be used as the basis for an FPGA- or even CPLD-based Alpha Juno oscillator clone. Because it uses a phase accumulator and a relatively low sample rate, the output from this code aliases quite badly at even fairly low pitches, especially with the "3 Oct Up" pulses in action. However it does quite closely resemble an Alpha Juno sound, and could be developed further. + + + diff --git a/alphaOsc.jpg b/alphaOsc.jpg new file mode 100644 index 0000000000000000000000000000000000000000..d27a4d8672b11b5d19e4604fca19c326cdb63a72 GIT binary patch literal 9692 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 UUH+wEaO3<=3ZMtWbMI#V3uwT8(EtDd literal 0 HcmV?d00001