/opl3_fpga

Reverse engineered SystemVerilog RTL version of the Yamaha OPL3 (YMF262) FM Synthesizer

Primary LanguageVerilogGNU Lesser General Public License v3.0LGPL-3.0

opl3_fpga

Reverse engineered SystemVerilog RTL version of the Yamaha OPL3 (YMF262) FM Synthesizer. Design is complete and working on the Digilent ZYBO board. Further testing is needed to verify full functionality and coverage. I'll mostly be adding and working on the software at this point.

See it in action:

There are some minor differences between this version and the original chip, mainly due to the hardware on the board that I'm using. The design is targeted to the Digilent ZYBO board which has the Xilinx Zynq-7000 SoC containing an ARM dual-core Cortex-A9 and an FPGA. The board also has an Analog Devices SSM2603 audio codec with dual 24-bit DACs. So the interfaces are different--the interface to the CPU is AXI4-Lite and the interface to the DAC is I2S, matching the particular hardware I have to work with. These interfaces are wrapped in the design and would be easy to swap out.

The original OPL3 chip used a 14.31818MHz master clock. The sample rate was 14.31818MHz/288 = 49.7159KHz. With 36 operator slots, that gives 8 clocks to update each operator each sample (operator logic is time-shared between slots).

The SSM2603 on the ZYBO uses 256x oversampling, and thus requires a master clock 256x the sample clock. I decided to use a single clock domain and keep the sample clock as close as possible to the original chip. The ZYBO provides the FPGA with an external clock of 125MHz. Using a mixed-mode clock manager (MMCM) in the FPGA, I'm able to synthesize a 12.727MHz clock from that, which divided by 256 gets me very close to the original sample rate at 49.7148KHz. Updating 36 operator slots using my slower master clock only gives me 7 clock cycles per sample instead of 8, but that's okay because I can stack tons of combinational logic up with few pipeline registers with this slow clock speed and modern FPGA.

Several other very smart people put in a lot of work before me to get as close as possible to bit-true software implementations of the chip, and this FPGA design would not be possible without their work. They're all over at http://forums.submarine.org.uk/phpBB/viewforum.php?f=9

Tools used are Modelsim, Vivado 2015.1, Octave (for sample analysis), and SVEditor (for SystemVerilog file editing).

Current utilization in xc7z010:

+----------------------------+-------+-------+-----------+-------+
|          Site Type         |  Used | Fixed | Available | Util% |
+----------------------------+-------+-------+-----------+-------+
| Slice LUTs                 |  7930 |     0 |     17600 | 45.06 |
|   LUT as Logic             |  7864 |     0 |     17600 | 44.68 |
|   LUT as Memory            |    66 |     0 |      6000 |  1.10 |
|     LUT as Distributed RAM |     0 |     0 |           |       |
|     LUT as Shift Register  |    66 |     0 |           |       |
| Slice Registers            | 11083 |     0 |     35200 | 31.49 |
|   Register as Flip Flop    | 11083 |     0 |     35200 | 31.49 |
|   Register as Latch        |     0 |     0 |     35200 |  0.00 |
| F7 Muxes                   |   368 |     0 |      8800 |  4.18 |
| F8 Muxes                   |     0 |     0 |      4400 |  0.00 |
+----------------------------+-------+-------+-----------+-------+

+-------------------+------+-------+-----------+-------+
|     Site Type     | Used | Fixed | Available | Util% |
+-------------------+------+-------+-----------+-------+
| Block RAM Tile    |    1 |     0 |        60 |  1.67 |
|   RAMB36/FIFO*    |    0 |     0 |        60 |  0.00 |
|   RAMB18          |    2 |     0 |       120 |  1.67 |
|     RAMB18E1 only |    2 |       |           |       |
+-------------------+------+-------+-----------+-------+

+----------------+------+-------+-----------+-------+
|    Site Type   | Used | Fixed | Available | Util% |
+----------------+------+-------+-----------+-------+
| DSPs           |    1 |     0 |        80 |  1.25 |
|   DSP48E1 only |    1 |       |           |       |
+----------------+------+-------+-----------+-------+

Serial command line interface

I've added a simple command line interface for playing songs. Set your terminal to 115200 baud, 8-N-1.

Welcome to the OPL3 FPGA

Type 'help' for a list of commands
>ls
descent.dro 381376
doom_000.dro 73778
doom_001.dro 54434
doom_002.dro 81288
doom_035.dro 50934
doom_036.dro 58198
doom2_000.dro 16554
doom2_001.dro 74836
doom2_002.dro 115962
doom2_003.dro 162264
doom2_031.dro 45292
doom2_032.dro 54584
doom2.dro 47328
doom.dro 90077
duke3d.dro 136862
hexen.dro 122449
war_000.dro 120658
>play doom_000.dro
DRO 2.0 file 

Build/run instructions (Linux)

  1. If you want to add any .dro files, you may place them in software/opl3dro. They will be included in the in-memory filesystem for playback.

  2. Source the Vivado and SDK settings so all the build tools are in your path. Example:

     source /opt/Xilinx/Vivado/2015.1/settings64.sh
     source /opt/Xilinx/SDK/2015.1/settings64.sh
    
  3. Run 'make' to build the FPGA and software necessary to run the OPL3 and create an SD card image.

  4. Copy the resulting BOOT.bin to an SD card, insert it into the ZYBO.

  5. Set JP5 to SD.

  6. Connect a USB cable to PROG/UART on the ZYBO. Power on the ZYBO. Your PC should detect the USB serial port device.

  7. Run a terminal program, use 115200 baud, 8-N-1.

  8. Reset the ZYBO by pressing PS-SRST. In your terminal you should see:

     Welcome to the OPL3 FPGA
    
     Type 'help' for a list of commands
     >
    

Enjoy!