# Field Programmable Digital Audio Effects Rack Andrew B. Shapiro Marc P. Resnick 6.111 Final Project, Fall 2010 #### Abstract This report presents the design and implementation for a Field Programmable Digital Audio Effects Rack using the Xilinx 2 series FPGA. With an 18-bit AC'97-compatible codec (audio) and a 24-bit VGA output (video), the FPGA can be used to synthesize the large, even cumbersome devices normally used by musicians and sound engineers to modify sound in real-time. Common audio effect tools such as signal generators, delay, and filters will be available to the user. Modular effects blocks and configurable dataflow allow for an extensible library and numerous arrangements of effects. A routing module allows for the use of a patch bay to configure the signal path, and a video display shows the activity of each effect's input. The final result is a modular system for emulation of audio effect devices many times the size of an FPGA which is compatible and interchaneable with the devices it emulates. # Contents | 1 | Intr | roduction | 4 | | |--------------|----------------------------|-----------------------------------------|----|--| | 2 | Ove | erview | 4 | | | | 2.1 | Effects | 5 | | | | 2.2 | Routing | 6 | | | | 2.3 | Display | 6 | | | | 2.4 | Composition and Extensions | 6 | | | 3 | Des | scription | 7 | | | | 3.1 | Patchbay (Drew) | 7 | | | | 3.2 | Routing (Drew) | 8 | | | | 3.3 | Effects | 8 | | | | | 3.3.1 Mix (Marc) | 8 | | | | | 3.3.2 Signal Generator (Marc) | 9 | | | | | 3.3.3 FIR Filter (Marc) | 9 | | | | | 3.3.4 Delay (Marc) | 9 | | | | | 3.3.5 Pan (Drew) | 11 | | | | 3.4 | Display (Drew) | 11 | | | | 3.5 | Demo Configuration | 11 | | | 4 | Test | ting and Troubleshooting | 11 | | | | 4.1 | Use of the Signal Generator for Testing | 12 | | | | 4.2 | Routing | 12 | | | | 4.3 | Delay and ZBT RAM | 12 | | | 5 | Cor | nclusions | 13 | | | 6 | Арр | pendices | 13 | | | $\mathbf{A}$ | Code Listing (alpha-order) | | | | | | A.1 | Main Audio Support (audio.v) | 14 | | | | A.2 | Router (audio_router.v) | 21 | | | | A.3 | Main Labkit File (audiofxbox.v) | 23 | | | | A.4 | Audio-Visual Feedback (channel_display.v) | 36 | | | | |-----------------|------|------------------------------------------------------------------------------------------------|----|--|--|--| | | A.5 | Delay (delay.v) | 38 | | | | | | A.6 | ZBT Suuport for Delay (delay_zbt.v) | 42 | | | | | | A.7 | FIR (filter.v) | 44 | | | | | | A.8 | Mix (mix.v) | 47 | | | | | | A.9 | Pan (pan.v) | 48 | | | | | | A.10 | Patchbay (patchbay.v) | 51 | | | | | | A.11 | Signal Generator (signal_generator.v) | 53 | | | | | | | | | | | | | List of Figures | | | | | | | | | 1 | Overview of Demonstration Configuration. LPF is an instance of a Low-Pass Filter. The | | | | | | | | effects such as Delay (D), Signal Generator (SG), and other Filters (FIR) receive their inputs | | | | | | | | from input_vector and pipe their ouputs into output_vector | 7 | | | | | | 2 | Sliding Window of ZBT Memory used by Delay | 10 | | | | | | | | | | | | ## 1 Introduction The Field Programmable Digital Audio Effects Rack is a system designed to emulate, replace, or work alongside a rack of dedicated audio effect devices. The patch bay and signal routing provide a familiar method of directing an audio signal through many effects. This implementation includes a signal generator, delay, filters, pan, and mix, allowing for both the modulation of sound input and synthesis of new sound. Additionally, multiple modules can be composed to create new effects, and the entire system is designed to be extensible; additional modules can easily be created and integrated into the system. ### Background Since the early 20th century, electronic music and instruments have risen in popularity. From the theremin, to the tape recorder, to the synthesizer, and so on, new musical techniques have accompanied technological advances. In order to integrate music with computers and other digital hardware, a technique called Digital Signal Processing (DSP) is used to manipulate audio in the form of quantized samples, rather than a continuous analog signal. Using an FPGA, the Field Programmable Digital Audio Effects Rack performs DSP to create and route electronic music effects. #### Motivation The Field Programmable Digital Audio Effect Rack emulates a "patch panel." This type of devices allows the user to reconfigure the order of effects in real time using a physical interface. Most patchbays literally connect the ports of separate audio modules such as the effects we have implemented for this project. However, we forego sending audio data into an external routing of physical cables and instead use a routing table to internally pass signals to the appropriate audio effects. This approach avoids the issue of noise due to cable or patchbay electrical imperfections, i.e., the click one would here when disconnecting the input to the speaker. ## 2 Overview The Audio Effects Rack can be divided into four main components: Audio Support, Audio Effects, Routing, and User Interaction. The **Audio Support** modules interact with an AC'97 codec chip which uses 18-bit resolution ADC's and DAC's; audio samples are converted from the audio input jack and produced at the at the audio output jack at a rate of 48kHz. The **Audio Effects** that this project implements are a foundational set of modules essential for an effective demonstration of the live routing features. The **Routing**-related modules orchestrate the updating and enforcing of routing configurations, and the **User Interaction** component allows audio channel activity to be visually monitored. #### 2.1 Effects The Audio Effects Rack implements several modules which synthesize and modulate sound based on parameters specific to each intended audio effect. All signals coming into and out of audio effects are signed 18-bit buses which can be interpretted as audio PCM samples or parameter settings. Sound effect primitives are composed to make higher-order blocks which may be more familiar to electronic musicians. #### Signal Generator The signal generator is the main producer of audio signals besides the microphone input. Lookup tables for sine, sawtooth, triangle, and square waves are used to provide a choice in the timbre of the produced sound. The frequency of all of these sounds is selectable by indicating a count which is used to increment the index into the appropriate look-up table. An optional input is also available which will be mixed equally with the generated signal (see Mix). #### Finite Input Response (FIR) Filter In order to attenuate unwanted frequencies the filter module applies a convolution of a 31-tap FIR filer with a historical buffer of input samples. This effect is not only useful in live performace, but doubles as an audio support module. Instances of this module are used on the main input and output signals to/from the AC'97 codec chip (see Figure 1) to counteract aliasing and reconstruction artifacts. Malformed filters can be used to generate irregular sounding transformations for more experimental sounds. #### Delay The delay introduces intentional latency to the signal path by buffering audio samples in ZBT memory. The size of this buffer and speed of playback are ideally paramterized, but have been set to constants for this demonstration. Delay is an useful effect for higher-order compsition of effects such as Chorus, Flanger, and Echo. ### 2.2 Routing Routing in the Audio Effects Rack is achieved using two components with a shared memory to store a routing table. The patchbay scans the physical interface and updates the routing table. The router uses the information in the routing table to pass input signals to the appropriate destination. #### **Patchbay** The patchbay uses a set of scanning and probing channels which are normally pulled up to VDD. To scan for active connections, a scan pattern of one bit on is shifted through scan channels. If there is a connection from Output 0 to Input 1, the FPGA-driven low signal on the Output 0 scan channel will be read at the probe channel corresponding to Input 1. All combinations of sources and destinations are tested continuously using the high and low halves of a 1kHz 8-bit counter. When a scan hit is sensed, the write enable for the routing table is asserted. In the case of a scan miss, the source channel set the highest patchbay address to indicate a no-connect. #### Router Information describing the connectivity of modules is stored in the Routing Table. This data is used on the rising edge of the AC'97s ready signal to synchronously assign slices of data from the input vector to the appropriate slices of the routing output vector. Generally, the outputs of the effects blocks are sent to the input vector of the routing table, and the inputs to the effects blocks are connected to slices of the output vector of the router. #### 2.3 Display A VGA display provides visual feedback of audio channel activity to aid in live performance. The values of input samples to effects blocks are displayed at 60Hz so that a signal that isn't necessirly heard can be visually checked by a performer. In addition to the volumetric displays, the routing table is also outputted to the 16-segment displays on the 6.111 labkit. #### 2.4 Composition and Extensions Each effect module is designed for easy composition with other effects. All input selectors have an 18-bit resolution, meaning that the output of any module can be used as a selector, and not just the signal input, to another module. For example, a sine wave from the signal generator can vary the buffer size of the delay module. The 18-bit standardization of all effects modules also allows for extensibility of the system as a whole. If the specifications set by the modules already implemented are followed, any new effect module can be implemented and added into the routing system. This design therefore represents more than simply a series of implemented digital effects, but rather a framework for the routing of a digital signal through any included modules. ## 3 Description Figure 1: Overview of Demonstration Configuration. LPF is an instance of a Low-Pass Filter. The effects such as Delay (D), Signal Generator (SG), and other Filters (FIR) receive their inputs from **input\_vector** and pipe their ouputs into **output\_vector**. ## 3.1 Patchbay (Drew) The patchbay is operated using two of the user ports on the lab kit as scan and probe channels. For the sixteen channel example we configured for the lab demonstration, an 8-bit counter is used to address every pair of input and output channels. Each cycle augments the test circuit, exposing the correct address, data, and write enable signals to the routing table. For each source address, a scan pattern with an active low signal on the given source channel is driven. Then each destination channel is probed to find if its value is zero (meaning a connection is present). The physical ports of the patchbay must be connected to pull-up resistors in order to ensure a high logic value throughout no-connect scenarios. The external user buses on the labkit also require a slower clock speed of 1kHz. These periodic updates of connectivity directly effect the internal routing table of the router module. ## 3.2 Routing (Drew) The routing provided by the Audio Effects Rack takes advantage of register/wire-arrays, input/ouput-vector slicing, and a lookup table for routing. The module receives all the outputs from the effects blocks as one concatenated vector of 18-bit audio samples (called the output vector). Similarly, the router exposes a concatenated vector of samples to go into the effects blocks (called the input vector). Generate blocks are used to assign 18-bit slices of the input and output vectors to wire and reg arrays which can be indexed by src/dest channel. To synchronously assign all outputs to their appropriate input channels, a for-loop iterates through all routing table address and performs the appropriate deferred assignment using the more convenient array data structure. This allows arbitrary routing changes from the patchbay to be applied seamlessly in the next 48kHz frame. #### 3.3 Effects #### 3.3.1 Mix (Marc) Mix allows for two input signals to be combined and output as one. An 18-bit input selector is used to calculate the relative weight of each signal in the combination. The signals are multiplied by the appropriate weight, then summed into the output signal. This module is especially useful when instantiated by other modules, providing for a "wet/dry" selector commonly found in audio effects devices. "Wet" and "dry" represent the modulated and unaffected input signals, respectively. #### 3.3.2 Signal Generator (Marc) The signal generator is capable of creating 1024-sample, 18-bit sine, square, sawtooth, and triangle waves at varying frequencies. The module takes a frequency and shape selector as in put, and produces the desired digital signal as its output. To generate the shapes, look-up tables for sine, sawtooth, and triangle waves are instantiated. The tables return the value of a sample for an index between 0 and 1023. The incrementation speed of the index is controlled by a counter that increments on every clock cycle. When the counter reaches a certain value, decided by the frequency select input, the index increments, and the counter resets. This allows the user to control the frequency of the output signal. The sine look-up table was generated in Xilinx ISE using coregen, and the triangle and sawtooth tables were created using MATLAB. The square shape is generated by combinational logic in the signal generator module. It is low for the first 512 samples, and high for the rest. A mix module is also instantiated by the signal generator in order to allow other signals to pass through it. Different sounds can then be chained together, to generate polyphonic tones. #### 3.3.3 FIR Filter (Marc) The Finite Impulse Response filter module attenuates and intensifies certain frequencies of an input signal given the filter coefficients of the desired frequency response. The output is the convolution of the coefficients with the input signal. An accumulator performs the convolution by adding incrementing by the product of a coefficient and sample on every clock cycle. An offset is used to step through the samples backwards, in order to properly calculate the convolved output. MATLAB was used to generate filter coefficients, and a python script was written to generate a case statement given a list of coefficients. The case statement is implemented as a ROM on the FPGA. Like the signal generator look-up tables, a coefficient is returned given an index. #### 3.3.4 Delay (Marc) Delay can best be described as a window that moves along a set of memory addresses with time. When a new sample is ready from the AC'97, it is written to the address specified by front end of the window, while the address at the back end of the window is read from and sent to the AC'97 for playback. Using this method, a sample which has been recorded into memory will be played back after the full length of the window has passed over it. Therefore, the delay time is controlled by the length of the window, specified by the number of samples in the window by an input to the module, as shown in Figure 2. For example, a window size of 48,000 samples results in a one second delay, since the sample rate is 48KHz. Figure 2: Sliding Window of ZBT Memory used by Delay. The labkit's onboard ZBT SRAM is used as the memory for the delay. In order to account for clock skew between the FPGA and memory modules, a helper module, provided by the 6.111 staff in 2005, was used to invert the clock to the SRAM. It also provided an easy interface to the RAM by allowing write data to be input and providing read data as output, even though the memory is single-port. A write-enable signal to this module pulls low the write-enable signal to the RAM, indicating that it should write the current data in the bus to the specified address. Otherwise, the memory returns the data in the specified address along the data bus. Each of these operations has a two-cycle delay. Therefore, the data from a read request cannot be latched until two cycles later, and the written data is not available in memory until two cycles after write functionality was enabled. To accomplish the reading, writing, and moving of the window along the memory, a finite state machine manages the module's current task. The state machine operates as follows: - 1. Do nothing until ready signal is asserted. - 2. On ready signal, write current sample to highest address in window. - 3. After write, read sample from lowest address in window. Wait two clock cycles, then latch data. - 4. Latched data is next sample out from module. Wait again for ready signal. A mix module is also instantiated to allow for both the original input sample and the delayed output sample to be output, allowing for a "wet/dry" signal selector to be used. #### 3.3.5 Pan (Drew) The pan module duplicates a mono signal and weights left and right output channels to reflect the currently select pan value. The transfer curve for pan is equal-power which means that the gain applied to the stereo channel does not grow linearly. To achieve this equal-power curve, a 16-bin lookup table is used. The appropriate gain coefficient is selected for left and right channels, and the new weighted values are output to the left and right output channels ## 3.4 Display (Drew) The display is a simple feedback mechanism for the potential musician who would use the Audio Effects Rack. The DCM is used to generate a buffer 65MHz clock which is used as a pixel clock with an xvga module (this builds the appropriate control signals for the VGA display). Each channel's input is latched at 60Hz and displayed as a vertical value bar of a color matching a patchbay channel. Appropriate spacing is coded for the given modules setup during the Demo Configuration. Due to undersampling and the resolution of the current video setup, this display is mostly useful for check alive-or-dead activity on a given channel (e.g., microphone in). ## 3.5 Demo Configuration The labkit setup for the demo was designed for a live performance. We included 4 signal generators (with selected shape mapped to the labkit switched) of frequencies in an A Major chord. Each one of the signal generators had a equal-gain mix passing upstream signals through to the next effect. Two FIR filters were also mapped to the patchbay to show the effects of not including the anti-aliasing and reconstruction filters. Finally a delay module was provided to utilize during feedback improvisation. The main in/out on the patchbay represented the live microphone samples and the outgoing speaker samples. Using all of these effects together made for an entertaining demostration of patchable sounds. ## 4 Testing and Troubleshooting All modules were initially tested with GTKwave andor ModelSim, using a Verilog test module that provided the appropriate clock, ready, and other input signals. However, simulation proved insufficient in several cases, most of which involved components that ModelSim could not simulate. Each module presented new challenges and issues that needed to be overcome for successful implementation. After simulation, modules were tested individually on the FPGA, and when they proved functional, were integrated with the rest of the system. ### 4.1 Use of the Signal Generator for Testing The signal generator was intentionally one of the first modules to be implemented. All of the available wave shapes are useful for testing audio-related modules, both to hear using the AC'97, and to see with the logic analyzer or modelsim. It therefore became the primary testing tool for most other modules. #### 4.2 Routing To develop a scheme for updating the routing table, many aproaches were explored in simulation. It be came clear that each pair of source and destination channels needed to be tested, so a simulation was first developed which would iterate through all possible pairs. This was achieved by the use of a counter that was twice as wide as an value in the routing table. This meant that the higher four bits could be regarded as a steady address while the lower four bits changed every clock cycle. Simulation also helped confirm that syntax for slicing the input and output vectors was working correctly. There were generally two test cases for the Patchbay simulations: (1) every channel n was fed into channel (Max - n), or (2) only one connection existed between the out of channel 4 and the in of channel 2. These scenarios were all that needed to be confirmed in order to expect functional behavior. Unfortunately, the unforseen circumstance of run an external bus faster than it can be read is not revealed through simulation. Only once the program was running on the FPGA did we notice off-by-one and off-by-two errors which symptoms of such a timing problem #### 4.3 Delay and ZBT RAM The delay effect experience several phases of memory management before settling on the use of the onboard ZBT SRAM. Initially, it was written to operate using coregen-created dual port BRAM. However, the synthesis of the amount of memory desired (2<sup>18</sup> bits) was taking almost an hour to complete. The same was apparently true when creating an inferred amount of BRAM using arrays of registers. Therefore, it was decided that the delay's data would be located in ZBT RAM. The ZBT RAM presented several new obstacles to be surmounted in order to have a working delay. Many of these were taken care of by the ZBT helper module provided by the 6.111 2005 website. It inverts the clock to the ZBT RAM in order to insure that data is held for a sufficient period of time. The main issue with interfacing with the memory was latching data at the correct time. Read requests to the RAM do not yield data until two clock cycles later. It was therefore necessary for the delay's state machine to use a counter to remain in its read state until it read data from the correct address. The other major problem involved some of the signals that needed to be held low in order for the RAM to function properly. After realizing that I had not assigned some of them to a value, it became apparent that this was the problem, and the delay began working soon after I corrected it. ## 5 Conclusions The Field Programmable Digital Audio Effects Rack is a modular, extensible system designed to integrate many audio effects easily into one device. The provided, on-the-fly signal routing system provides a familiar method for choosing and arranging effects, and the high sample rate and resolution are aimed at preserving sound quality. The FPGA provides a unique angle from which to approach an audio effects rack such as this one. It can, for instance, be thought of as a series of pre-programmed digital effects, with a provided routing system. We prefer to think of it as a framework that provides a musician or developer with an environment for managing digital effects on an FPGA, routing signals, and synthesizing new sounds. ## 6 Appendices ## A Code Listing (alpha-order) ## A.1 Main Audio Support (audio.v) ``` // 3 // bi-directional monaural interface to AC97 augmented by Drew Shapiro 5 // TODO: Make input channel binaural and (18-bit) [11/28/2010] // 9 module audio_support (/*AUTOARG*/ // Outputs ready\;,\;\;mono\_channel\_in\;,\;\;ac97\_sdata\_out\;,\;\;ac97\_synch\;,\;\;audio\_reset\_b\;, 11 // Inputs reset, clock_27mhz, volume, left_channel_out, right_channel_out, 13 ac97_sdata_in, ac97_bit_clock 15 ); input reset, clock_27mhz; input [4:0] volume; input [17:0] left_channel_out, right_channel_out; input ac97_sdata_in , ac97_bit_clock; //ac97 interface signal 19 21 output ready; output [17:0] mono_channel_in; 23 output ac97_sdata_out, ac97_synch; //ac97 interface signal reg audio_reset_b; //ac97 interface signal output 25 wire [7:0] command_address; wire [15:0] command_data; 27 command_valid; wire [19:0] left_in_data, right_in_data; 29 wire [19:0] left_out_data, right_out_data; 31 // wait a little before enabling the AC97 codec reg [9:0] reset_count; 33 always @(posedge clock_27mhz) begin if (reset) begin 35 ``` ``` audio_reset_b = 1'b0; 37 reset\_count = 0; end else if (reset_count == 1023) audio_reset_b = 1'b1; 39 else reset_count = reset_count +1; 41 end 43 wire ac97_ready; 45 ac97 ac97 (.ready(ac97_ready), .command_address(command_address), 47 .command_data(command_data), .command_valid(command_valid), .left_data(left_out_data), .left_valid(1'b1), 49 .right_data(right_out_data), .right_valid(1'b1), .left_in_data(left_in_data), .right_in_data(right_in_data), 51 .ac97_sdata_out(ac97_sdata_out), .ac97_sdata_in(ac97_sdata_in), 53 .ac97_synch(ac97_synch), .ac97_bit_clock(ac97_bit_clock)); 55 57 // ready: one cycle pulse synchronous with clock_27mhz reg [2:0] ready_sync; 59 always @ (posedge clock_27mhz) ready_sync <= {ready_sync[1:0], ac97_ready}; ready = ready_sync[1] & ready_sync[2]; assign 61 reg [17:0] l_left_data, l_right_data; //latch incoming audio data 63 always @ (posedge clock_27mhz) if (ready) begin 65 l_left_data <= left_channel_out;</pre> l_right_data <= right_channel_out;</pre> 67 69 assign mono_channel_in = left_in_data[19:2]; //FIX - monaural assign left_out_data = {l_left_data, 2'b00}; assign \ right\_out\_data = \{l\_right\_data , \ 2'b00\}; 71 73 //\ generate\ repeating\ sequence\ of\ read/writes\ to\ AC97\ registers ac97commands cmds(.clock(clock_27mhz), .ready(ready), ``` ``` 75 .command_address(command_address), .command_data(command_data), .command_valid(command_valid), 77 .volume(volume), // mic 79 .source(3'b000)); endmodule // audio 81 // assemble/disassemble AC97 serial frames 83 module ac97 ( output reg ready, input wire [7:0] command_address, 85 input wire [15:0] command_data, input wire command_valid, input wire [19:0] left_data, input wire left_valid, input wire [19:0] right_data, 91 input wire right_valid, output reg [19:0] left_in_data, right_in_data, output reg ac97_sdata_out, input wire ac97_sdata_in, output reg ac97_synch, input wire ac97_bit_clock 97); reg [7:0] bit_count; 99 reg [19:0] l_cmd_addr; 101 reg [19:0] l_cmd_data; reg [19:0] l_left_data, l_right_data; reg l_cmd_v , l_left_v , l_right_v ; 103 105 initial begin ready <= 1'b0; // synthesis attribute init of ready is "0"; 107 ac97\_sdata\_out <= 1'b0; 109 // synthesis attribute init of ac97_sdata_out is "0"; ac97_synch <= 1'b0; 111 //\ synthesis\ attribute\ init\ of\ ac97\_synch\ is\ "0"; bit_count <= 8'h00; 113 ``` ``` // synthesis attribute init of bit_count is "0000"; 115 l_cmd_v <= 1'b0; // synthesis attribute init of l_cmd_v is "0"; 117 l_left_v <= 1'b0; // synthesis attribute init of l_left_v is "0"; l_right_v <= 1'b0; 119 // synthesis attribute init of l_right_v is "0"; 121 left_in_data <= 20'h00000; // synthesis attribute init of left_in_data is "00000"; 123 right_in_data \le 20'h00000; // synthesis attribute init of right_in_data is "00000"; 125 end 127 always @(posedge ac97_bit_clock) begin 129 // Generate the sync signal if (bit\_count == 255) 131 ac97-synch \leq 1'b1; if (bit\_count == 15) 133 ac97\_synch <= 1'b0; // Generate the ready signal 135 if (bit_count == 128) ready <= 1'b1; 137 if (bit\_count == 2) ready <= 1'b0; 139 // Latch user data at the end of each frame. This ensures that the 141 // first frame after reset will be empty. if (bit_count == 255) begin 143 l_cmd_addr <= {command_address, 12'h000};</pre> l_cmd_data <= {command_data, 4'h0};</pre> 145 l_cmd_v <= command_valid;</pre> l_left_data <= left_data;</pre> 147 l_left_v <= left_valid; 149 l_right_data <= right_data;</pre> l_right_v <= right_valid;</pre> 151 end ``` ``` if ((bit_count >= 0) && (bit_count <= 15))</pre> 153 // Slot 0: Tags case (bit_count[3:0]) 155 4'h0: ac97_sdata_out <= 1'b1; // Frame valid // Command address valid 4'h1: ac97\_sdata\_out \le l\_cmd\_v; 157 4'h2: ac97\_sdata\_out \le l\_cmd\_v; // Command data valid 4'h3: ac97\_sdata\_out \le l\_left\_v; // Left data valid 159 4'h4: ac97_sdata_out <= l_right_v; // Right data valid 161 default: ac97_sdata_out <= 1'b0; endcase else if ((bit_count >= 16) && (bit_count <= 35)) 163 // Slot 1: Command address (8-bits, left justified) 165 ac97\_sdata\_out <= l\_cmd\_v \ ? \ l\_cmd\_addr[35-bit\_count] \ : \ 1'b0; else if ((bit_count >= 36) && (bit_count <= 55)) 167 // Slot 2: Command data (16-bits, left justified) ac97_sdata_out <= l_cmd_v ? l_cmd_data[55-bit_count] : 1'b0; 169 else if ((bit_count >= 56) && (bit_count <= 75)) begin // Slot 3: Left channel ac97_sdata_out <= l_left_v ? l_left_data[19] : 1'b0; 171 l_left_data <= { l_left_data[18:0], l_left_data[19] };</pre> 173 else if ((bit_count >= 76) && (bit_count <= 95)) 175 // Slot 4: Right channel ac97_sdata_out <= l_right_v ? l_right_data[95-bit_count] : 1'b0; 177 else ac97\_sdata\_out <= 1'b0; 179 bit\_count \le bit\_count+1; end // always @ (posedge ac97_bit_clock) 181 183 always @(negedge ac97_bit_clock) begin if ((bit_count >= 57) && (bit_count <= 76)) // Slot 3: Left channel 185 left_in_data <= { left_in_data[18:0], ac97_sdata_in };</pre> else if ((bit_count >= 77) && (bit_count <= 96)) 187 // Slot 4: Right channel 189 right_in_data <= { right_in_data[18:0], ac97_sdata_in }; end 191 endmodule ``` ``` 193 // issue initialization commands to AC97 module ac97commands ( 195 input wire clock, input wire ready, output wire [7:0] command_address, 197 output wire [15:0] command_data, 199 output reg command_valid , input wire [4:0] volume, input wire [2:0] source 201 ); 203 reg [23:0] command; 205 reg [3:0] state; initial begin 207 command \le 4'h0; // synthesis attribute init of command is "0"; 209 command_valid <= 1'b0;</pre> // synthesis attribute init of command_valid is "0"; state <= 16'h0000; 211 // synthesis attribute init of state is "0000"; 213 \mathbf{end} 215 assign command_address = command[23:16]; assign command_data = command[15:0]; 217 wire [4:0] vol; assign vol = 31-volume; // convert to attenuation 219 always @(posedge clock) begin 221 if (ready) state <= state+1;</pre> 223 case (state) 4'h0: // Read ID 225 begin command \le 24'h80_0000; 227 command_valid <= 1'b1;</pre> 229 end 4'h1: // Read ID ``` ``` 231 command \le 24'h80_0000; 4'h3: // headphone volume command <= \{ \ 8'h04, \ 3'b000, \ vol, \ 3'b000, \ vol \}; 233 4'h5: // PCM volume command \le 24'h18_0808; 235 4'h6: // Record source select 237 command <= \{ \ 8'h1A, \ 5'b00000, \ source, \ 5'b00000, \ source \}; 4'h7: // Record gain = max 239 command \leq 24'h1C_0F0F; 4\,{}^{\backprime}\mathrm{h9}\colon\ //\ \mathit{set}\ +20\,\mathit{db}\ \mathit{mic}\ \mathit{gain} 241 command \le 24'h0E_8048; 4'hA: // Set beep volume 243 command <= \ 24\,{}^{\backprime}h0A\,{}_{-}0000\,; 4\,{\rm 'hB}\colon\ //\ \mathit{PCM}\ \mathit{out}\ \mathit{bypass}\ \mathit{mix1} 245 command \le 24'h20_8000; default: 247 command \leq 24'h80_-0000; endcase // case(state) 249 end // always @ (posedge clock) {\bf end module} \ \ // \ \ ac 97 commands ``` ## A.2 Router (audio\_router.v) ``` 2 // // Audio Router for \ Field-Programmable \ Audio \ Effect \ Rack author:\ Drew\ Shapiro // 6 // 8 module audio_router(/*AUTOARG*/ // Outputs 10 block_input_vector, current_routing, // Inputs reset\ ,\ clk\ ,\ ready\ ,\ block\_output\_vector\ ,\ route\_clk\ ,\ route\_data\ ,\ route\_address\ , update_routing ); 14 parameter WIDTH = 18; 16 parameter LOGN = 4; localparam N = 1 \ll LOG.N; 18 input reset , clk , ready; 20 //accept a concatenated bus of output samples input [WIDTH*N-1:0] block_output_vector; wire [WIDTH-1:0] rcv_from_block [N-1:0]; //provide a concatenated bus of input samples output [WIDTH*N-1:0] block_input_vector; reg [WIDTH-1:0] snd_to_block[N-1:0]; 26 //maintain a routing table which can be updated externally input route_clk , update_routing; input [LOG_N-1:0] route_data, route_address; 30 output [LOG_N*N-1:0] current_routing; reg [LOG_N-1:0] routing_table [N-1:0]; 32 //slice input and output vectors 34 generate genvar i; 36 for(i=0;i<\!\!N;i=i+1) begin: slice ``` ``` \mathbf{assign} \ \, \mathbf{rcv\_from\_block} \, [\, \mathbf{i} \, ] \, = \, \mathbf{block\_output\_vector} \, [\, (\, \mathbf{i} \, + \! 1) \, * \! \mathbf{WIDTH-1:} \, \mathbf{i} \, * \! \mathbf{WIDTH-1:} \, \mathbf{i} \, * \! \mathbf{WIDTH-1:} \, \mathbf{i} \, * \! \mathbf{WIDTH-1:} \, \mathbf{i} \, * \! \mathbf{WIDTH-1:} \, \mathbf{i} \, * \! \mathbf{WIDTH-1:} \, \mathbf{i} \, * \, \mathbf{i} \, \mathbf{i} \, \mathsf{i} \mathbf{assign} \ \ \mathsf{block\_input\_vector}\left[(\ \mathsf{i} + 1) * \mathsf{WIDTH} - 1 : \mathsf{i} * \mathsf{WIDTH}\right] \ = \ \mathsf{snd\_to\_block}\left[\ \mathsf{i}\ \right]; 38 \textbf{assign} \ \, \texttt{current\_routing} \, [\, (\, \texttt{i} + 1) * \texttt{LOG.N} - 1 : \texttt{i} * \texttt{LOG.N}] \, = \, \texttt{routing\_table} \, [\, \texttt{i} \, ] \, ; 40 \mathbf{end} {\bf endgenerate} 42 //maintain a routing table which can be updated externally 44 always @ (posedge route_clk) if(update_routing) routing_table[route_address] <= route_data;</pre> 46 //enforce routing table 48 integer j; always @ (posedge clk) begin 50 for(j=0; j<\!\!N; j=j+1) begin snd_to_block[j] <= rcv_from_block[routing_table[j]]; // handle no-connects?</pre> 52 end \mathbf{end} 54 endmodule ``` ## A.3 Main Labkit File (audiofxbox.v) ``` 'timescale 1ns / 1ps 4 // Audio Effects Box // module audiofxbox( // AC97 8 output wire beep, audio_reset_b, ac97_synch, ac97_sdata_out, 10 input wire ac97_bit_clock, ac97_sdata_in, // VGA 12 14 output wire [7:0] vga_out_red, vga_out_green, vga_out_blue, output wire vga_out_sync_b, vga_out_blank_b, vga_out_pixel_clock, vga_out_hsync, vga_out_vsync, 16 // PS2 inout wire mouse_clock, mouse_data, keyboard_clock, keyboard_data, 18 20 // FLUORESCENT DISPLAY output wire disp_blank, disp_clock, disp_rs, disp_ce_b, disp_reset_b, 22 input wire disp_data_in, output wire disp_data_out, 24 // BUTTONS, SWITCHES, LEDS 26 //input wire button0, //input wire button1, 28 //input wire button2, //input wire button3, 30 //input wire button_enter, //input wire button_right, 32 //input wire button_left, //input wire button_down, //input wire button_up, 34 input wire [7:0] switch, 36 output wire [7:0] led, ``` ``` // USER CONNECTORS, DAUGHTER CARD, LOGIC ANALYZER 38 inout wire [11:0] user1 , inout wire [31:0] user2, 40 inout wire [31:0] user3, 42 inout wire [31:0] user4, //inout\ wire\ [43:0]\ daughtercard , 44 output wire [15:0] analyzer1_data, output wire analyzer1_clock, //output\ wire\ [15:0]\ analyzer2\_data\ ,\ output\ wire\ analyzer2\_clock\ , output wire [15:0] analyzer3_data, output wire analyzer3_clock, 46 output wire [15:0] analyzer4_data, output wire analyzer4_clock, input wire clock_27mhz, 48 50 // ZBT RAM inout [35:0] ram0_data, output [18:0] ram0_address, 52 output ram0_adv_ld, output ram0_clk, ram0_cen_b, 54 output ram0_ce_b, ram0_oe_b, 56 output ram0_we_b, output [3:0] ram0_bwe_b 58 ); 60 // ZBT Signals 62 // 64 assign ram0\_ce\_b = 0; 66 assign ram0_oe_b = 0; assign ram0_adv_ld = 0; 68 assign ram0_bwe_b = 3'b0; 70 72 // 74 // Timing and Reset // ``` ``` 76 wire clock_65mhz_unbuf, clock_65mhz; 78 // use FPGA's digital clock manager to produce a // 65MHz clock (actually 64.8MHz) 80 DCM vclk1 (.CLKIN(clock_27mhz),.CLKFX(clock_65mhz_unbuf)); 82 // synthesis attribute CLKFX_DIVIDE of vclk1 is 10 // synthesis attribute CLKFX_MULTIPLY of vclk1 is 24 // synthesis attribute CLK_FEEDBACK of vclk1 is NONE 84 // synthesis attribute CLKIN_PERIOD of vclk1 is 37 BUFG vclk2(.O(clock_65mhz),.I(clock_65mhz_unbuf)); 86 88 // power-on reset generation // remain high for first 16 clocks wire power_on_reset; SRL16 reset_sr (.D(1'b0), .CLK(clock_65mhz), .Q(power_on_reset), .A0(1'b1), .A1(1'b1), .A2(1'b1), .A3(1'b1)); 92 defparam reset_sr.INIT = 16'hFFFF; //divide down a 10Hz clock for visual display reg clock_10hz, clock_1khz; 96 reg [20:0] count_20hz; reg [13:0] count_2khz; assign led = \{8\{\operatorname{clock\_10hz}\}\}; always @ (posedge clock_27mhz) begin 100 if(power_on_reset) begin \operatorname{clock}_{-10hz} \ll 1; 102 \operatorname{clock}_{-1}\operatorname{khz} \iff 1; count_20hz \le 0; 104 count_2khz \ll 0; end 106 else begin clock_10hz \le ((count_20hz+1) = 1.350.000)? clock_10hz: clock_10hz; count_20hz \le ((count_20hz+1) = 1_350_000) ? 0 : count_20hz+1; 108 clock_1khz <= ((count_2khz+1) == 13_500) ? ~clock_1khz : clock_1khz; 110 count_2khz \le ((count_2khz+1) = 13.500) ? 0 : count_2khz+1; 112 end end 114 ``` ``` // 1 MHz Clock for ADC one_mhz_clock one_mhz (.clock_27mhz(clock_27mhz), .clock_1mhz(clock_1mhz)); 116 assign user2[18] = clock_1mhz; wire [7:0] knob_value; 118 assign knob_value = user2[26:19]; 120 122 // // Audio Support 124 wire ready; 126 //assign beep = power_on_reset; 128 wire [17:0] mono_channel_in, left_channel_out , right_channel_out , left_channel_raw , right_channel_raw , raw_in; 130 wire [27:0] antialias_out, antiimage_out1, antiimage_out2; 132 //assign\ left\_channel\_out = left\_channel\_raw; //assign\ right\_channel\_out = right\_channel\_raw; 134 //filter 64 antialias (.clock(clock_27mhz), reset(reset), ready(ready), x(left\_channel\_raw) ,.y(left\_channel\_out)); 136 audio_support basic_audio_interface ( .reset (reset), .clock_27mhz(clock_27mhz), 138 .ready(ready), 140 .mono_channel_in(mono_channel_in), .left_channel_out(left_channel_out), 142 .right_channel_out(right_channel_out), .ac97_bit_clock(ac97_bit_clock), 144 .ac97_sdata_in(ac97_sdata_in), .ac97_sdata_out(ac97_sdata_out), 146 .ac97_synch(ac97_synch), .audio_reset_b (audio_reset_b), .volume(5'b11111) 148 ); 150 152 ``` ``` // 154 // Audio Effects 156 wire [16*18-1:0] input_vector, output_vector; 158 //Channel 0: Main I/O 160 wire [17:0] main_out_mono; 162 assign left_channel_out = main_out_mono; assign right_channel_out = main_out_mono; 164 assign main_out_mono = input_vector[17:0]; 166 168 //Channel 1: Delay 170 wire [17:0] delay_signal; delay delay1 ( . clock(clock_27mhz), 172 .reset (reset), 174 .ready(ready), .sample_in(input_vector[18*2-1:18]), .buffer_size(19'd24000), 176 .mix_out(delay_signal), .ram_clk(ram0_clk), 178 .ram_we_b(ram0_we_b), 180 .ram_address(ram0_address), .ram_data(ram0_data), .ram_cen_b(ram0_cen_b), 182 .mix_select(0) 184 ); //Channel 2: FIR 1 186 wire [17:0] fir1_signal; wire [27:0] fir1_signal_full; 188 \mathbf{assign} \ \ \mathsf{fir1\_signal} \ = \ \mathsf{fir1\_signal\_full} \left[ \, 2\,7\,{:}\,10 \, \right]; filter fir1 ( 190 .clock(clock_27mhz), ``` ``` 192 .reset (reset), .ready(ready), 194 .x(input\_vector[18*3-1:18*2]), .y(fir1_signal_full) 196 ); 198 //Channel 3: FIR 2assign wire [17:0] fir2_signal; 200 //Channel 4: Wave 1 A440 - 61 363 counts 202 reg [17:0] wave1_count; wire [17:0] wave1_signal; 204 signal_generator wavel ( .freq_select(18'd61_363), 206 .clock(clock_27mhz), .reset(reset), .wave_in(input_vector[18*5-1:18*4]), 208 .wave_out(wave1_signal), .shape(switch[1:0]) 210 ); //Channel 5: Wave 2 C#550 - 49 090 counts 212 reg [17:0] wave2_count; 214 wire [17:0] wave2_signal; signal_generator wave2 ( .freq_select(18'd49_090), 216 .clock(clock_27mhz), .reset(reset), .wave_in(input_vector[18*6-1:18*5]), 218 .wave_out(wave2_signal), .shape(switch[3:2]) 220 ); 222 //Channel 6: Wave 3 E660 - 40 909 counts reg [17:0] wave3_count; 224 wire [17:0] wave3_signal; 226 signal_generator wave3 ( . freq_select(18'd40_909), 228 .clock(clock_27mhz), .\,\,\mathrm{reset}\,(\,\mathrm{reset}\,)\;,\;\;.\,\,\mathrm{wave\_in}\,(\,\mathrm{input\_vector}\,[\,18*7\,-1{:}18*6\,]\,)\;, .wave_out(wave3_signal), 230 ``` ``` .shape(switch[5:4]) 232 ); 234 //Channel 7: Wave 4 A880 - 30 681 counts reg [17:0] wave4_count; wire [17:0] wave4_signal; 236 signal_generator wave4 ( 238 .freq_select(18'd30_681), . clock (clock_27mhz), . reset (reset), . wave_in(input_vector[18*8-1:18*7]), 240 .wave_out(wave4_signal), 242 .shape(switch[7:6]) ); 244 246 248 reg [17:0] select; 250 wire signed [17:0] mix_signal; always @ (posedge clock_10hz) select <= reset ? 0 : select + 15'b1000000000000000; 252 mix mix_fifth ( 254 .reset (reset), .clock(clock_27mhz), 256 .ready(ready), 258 .mix_select(0), .signal1 (wave1_signal), .signal2(wave2_signal), 260 .mixed(mix_signal) 262 ); wire signed [17:0] select_knob = {user1 [11:0], 6'b0}; 264 wire signed [17:0] right_panned, left_panned; 266 pan pan_main_out ( .clk(clock_27mhz),.reset(reset), 268 .ready(ready), ``` ``` 270 .pan_select(select_knob), .input_signal(main_out_mono), 272 .left_signal_out(left_panned), .right_signal_out(right_panned) 274 ); 276 assign left_channel_raw = switch[7] ? left_panned : main_out_mono; 278 assign \ right\_channel\_raw = switch [7] \ ? \ right\_panned : main\_out\_mono; 280 // 282 // Patchbay and Audio Routing 284 286 wire update_routing; wire [3:0] src, dest; wire [63:0] data; 288 290 292 wire [14*18-1:0] nothing; assign output_vector = {0, wave4_signal, wave3_signal, wave2_signal, wave1_signal, // 294 Waves fir2_signal, fir1_signal, // Filters delay_signal , // Delay 296 mono_channel_in }; // Line-In 298 audio_router #(.LOG_N(4),.WIDTH(18)) router1 ( .block_input_vector(input_vector), 300 .current_routing(data), 302 .reset (reset), .clk(clock_27mhz), .ready(ready), 304 . block_output_vector(output_vector), 306 .route_clk(clock_1khz), .route_data(src), ``` ``` 308 .route_address(dest), . update_routing(update_routing) 310 ); 312 //\ physical\ pathcbay\ updates\ routing\_table\ inside\ the\ router patchbay physical_interface ( 314 .scan_channels(user3[15:0]), .probe_channels(user4[15:0]), .update_routing(update_routing), 316 .src(src), . dest (dest), 318 .reset (reset), 320 .clock(clock_1khz) 322 assign user4 [31:8] = \{24\{1'b1\}\}; 324 326 // Keyboard and Mouse Input 328 330 wire [11:0] mouse_x, mouse_y; wire [2:0] mouse_click; ps2_mouse_xy mouse(.clk(clock_65mhz), .reset(reset), 332 .ps2_clk(mouse_clock), .ps2_data(mouse_data), .mx(mouse_x), .my(mouse_y), .btn_click(mouse_click)); 334 wire [2:0] command; 336 wire new_command; assign beep = new_command; 338 keyboard asdw_commands ( .clock_27mhz(clock_27mhz), 340 .reset (reset), .kbd_data(keybaord_data), 342 .kbd_clock(keyboard_clock), .command(command), 344 .new_command(new_command) ); 346 ``` ``` 348 // // VGA Display 350 352 354 // generate basic XVGA video signals wire [10:0] hcount; 356 wire [9:0] vcount; wire hsync, vsync, blank; xvga xvga1(.vclock(clock_65mhz),.hcount(hcount),.vcount(vcount), 358 .hsync(hsync),.vsync(vsync),.blank(blank)); wire [2:0] pixelmainout, pixelmainin, pixel1, pixel2, pixel3, pixel4, pixel5, pixel6, 360 pixel7, pixel8; 362 channel_display #(.LEFT(100),.COLOR(7)) main_out_display ( .vga_clock(clock_65mhz),.reset(reset), .pixel(pixelmainout), 364 . hcount(hcount),.vcount(vcount),.vsync(vsync),.value(main_out_mono)); 366 channel_display #(.LEFT(175),.COLOR(3)) 368 main_in_display ( .vga_clock(clock_65mhz),.reset(reset), .pixel(pixelmainin), 370 . hcount (hcount), .vcount (vcount), .vsync(vsync), .value(mono_channel_in)); 372 channel_display #(.LEFT(350),.COLOR(2)) wavel_display ( 374 .vga_clock(clock_65mhz),.reset(reset), .pixel(pixel1), . hcount (hcount), . vcount (vcount), . vsync(vsync), . value(input_vector[18*2-1:18])); 376 channel_display #(.LEFT(425),.COLOR(1)) 378 wave2_display ( .vga_clock(clock_65mhz),.reset(reset), .pixel(pixel2), . hcount(hcount),.vcount(vcount),.vsync(vsync),.value(input_vector[18*3-1:18*2])); 380 382 channel_display #(.LEFT(500),.COLOR(4)) wave3_display ( .vga_clock(clock_65mhz),.reset(reset), .pixel(pixel3), 384 ``` ``` . hcount (hcount),.vcount (vcount),.vsync(vsync),.value(input_vector[18*4-1:18*3])); 386 channel_display #(.LEFT(675),.COLOR(6)) 388 wave4_display ( .vga_clock(clock_65mhz),.reset(reset), .pixel(pixel4), 390 .\ hcount(hcount), .\ vcount(vcount), .\ vsync(vsync), .\ value(input\_vector[18*5-1:18*4])); 392 394 channel_display #(.LEFT(750),.COLOR(1)) 396 wave5_display ( .vga_clock(clock_65mhz),.reset(reset), .pixel(pixel5), . hcount (hcount), . vcount (vcount), . vsync(vsync), . value(input_vector[18*6-1:18*5])); 398 400 channel_display #(.LEFT(825),.COLOR(4)) wave6_display ( 402 .vga_clock(clock_65mhz),.reset(reset), .pixel(pixel6), . hcount (hcount),.vcount (vcount),.vsync(vsync),.value(input_vector[18*7-1:18*6])); 404 406 channel_display #(.LEFT(900),.COLOR(2)) 408 wave7_display ( .vga_clock(clock_65mhz),.reset(reset), .pixel(pixel7), . hcount (hcount), .vcount (vcount), .vsync(vsync), .value(input_vector[18*8-1:18*7])); 410 412 reg [2:0] rgb; reg b, hs, vs; 414 always @(posedge clock_65mhz) begin hs <= hsync; 416 vs <= vsync; b <= blank; rgb <= ((hcount == mouse_x | vcount == mouse_y) ? 7 : 0) | 418 (pixelmainout | pixelmainin | pixel1 | pixel2 | pixel3 | pixel4 | pixel5 | pixel6 | pixel7 | pixel8 ) | 420 (switch [7] ? hcount [3:1] + vcount [3:1] : 0); 422 end ``` ``` // VGA Output. In order to meet the setup and hold times of the // AD7125, we send it \ ^{\circ} clock_65mhz. 424 assign vga_out_red = \{8\{rgb[2]\}\}; assign vga_out_green = \{8\{rgb[1]\}\}; 426 assign vga\_out\_blue = \{8\{rgb[0]\}\}; assign vga_out_sync_b = 1'b1; // not used 428 assign vga_out_blank_b = ~b; 430 assign vga_out_pixel_clock = ~clock_65mhz; assign vga_out_hsync = hs; 432 assign vga_out_vsync = vs; 434 // Hex Display 436 // 438 display_16hex disp16 (.reset (reset), .clock_27mhz (clock_27mhz), 440 .data(data), .disp_blank(disp_blank), .disp_clock(disp_clock), 442 .disp_rs(disp_rs), .disp_ce_b(disp_ce_b), .disp_reset_b (disp_reset_b), .disp_data_out (disp_data_out)); 444 assign analyzer1_clock = clock_65mhz; 446 assign analyzer1_data[3:0] = src; assign analyzer1_data[7:4] = dest; 448 assign analyzer1_data[8] = update_routing; 450 452 assign analyzer1_data[15:9] = \{7\{1'b1\}\}; 454 assign analyzer3_clock = clock_65mhz; assign analyzer3_data[15:0] = user3[15:0]; 456 458 assign analyzer4_clock = clock_65mhz; assign analyzer4_data[15:0] = user4[15:0]; 460 ``` endmodule ## A.4 Audio-Visual Feedback (channel\_display.v) ``` 3 // Channel Display // for Field Programmable Audio Effect Rack 5 // author: Drew Shapiro module channel_display(vga_clock, vsync, reset, pixel, hcount, vcount, value); parameter COLOR = 7; parameter LEFT = 100; parameter BOTTOM = 100: 11 parameter WIDTH = 20; parameter HEIGHT = 512; 13 localparam TOP = BOTTOM + HEIGHT; 15 input vga_clock; 17 input reset; input [17:0] value; wire [8:0] disp_value; 19 reg [17:0] l_value; //grab pieces of the hi-res reading for visual feedback assign disp_value [8:6] = {l_value [17], l_value [15], l_value [13]}; 21 assign disp_value [5:3] = \{l_value [11], l_value [9], l_value [7]\}; assign disp_value [2:0] = \{l_value[4], l_value[2], l_value[0]\}; output reg [2:0] pixel; input [10:0] hcount; 25 input [9:0] vcount; input vsync; reg [1:0] vsync_buffer; 29 wire frame_refresh; // only latch once every frame 31 assign frame_refresh = vsync_buffer[0] & ~vsync_buffer[1]; wire within_bounds, on_border; assign within_bounds = ( hcount >= LEFT & hcount < LEFT + WIDTH) & 33 ( vcount >= BOTTOM & vcount < BOTTOM + HEIGHT); assign on_border = within_bounds & ((hcount == LEFT) | (hcount == LEFT + WIDTH-1)) | 35 ((vcount = BOTTOM) | (vcount = BOTTOM + HEIGHT-1)); 37 ``` ``` {\bf always} \ @ \ ({\bf posedge} \ {\rm vga\_clock}\,) \ {\bf begin} 39 vsync_buffer <= {vsync_buffer[0], vsync};</pre> l_value <= frame_refresh ? value : l_value;</pre> 41 if(within\_bounds) begin \label{eq:pixel} {\tt pixel} \, <= \, ({\tt vcount} \, >= \, ({\tt BOTTOM} \, + \, 512 \, - \, {\tt disp\_value})) \, \, ? \, \, {\tt COLOR} \, : \, \, 0; \, \, /\!/ {\tt only} \, \, \, {\tt paint} \, \, {\tt COLOR} \, \, if \, \, \, {\tt only} \, \, {\tt paint} \, \, {\tt COLOR} \, \, if \, \, {\tt only} \, \, {\tt paint} \, \, {\tt COLOR} \, \, if \, \, {\tt only} \, \, {\tt paint} \, \, {\tt COLOR} \, \, if \, \, {\tt only} value visible 43 \quad \mathbf{end} \quad _{ m else} 45 pixel \le 0; \mathbf{end} 47 end module \\ ``` # A.5 Delay (delay.v) ``` // Delay module for 6.111 2010 Final Project 4 // Marc Resnick // 8 module delay (input clock, reset, ready, input signed [17:0] sample_in, input [18:0] buffer_size , input signed [17:0] mix_select , 10 output signed [17:0] mix_out, output ram_clk, output ram_we_b, output [18:0] ram_address, 12 inout [35:0] ram_data, output ram_cen_b); 14 parameter SAMPLE_RATE = 480000; 16 wire [35:0] write_data; wire [35:0] read_data; 18 reg [18:0] cur_addr; 20 wire [18:0] write_addr; 22 reg [17:0] sample_out; 24 we; 26 reg [1:0] rwstate = 2'b0; reg [1:0] next_rwstate = 2'b0; 28 wire [18:0] addr; 30 assign addr = we ? write_addr : cur_addr; //if write enabled, address is write address ( front of window), otherwise, it's read address (back of window) assign write_data = sample_in; 32 //write data always incoming sample 34 zbt_6111 delay_zbt ( //helper module for zbt ram 36 .clk(clock), ``` ``` .cen(1'b1), 38 .we(we), .addr(addr), .write_data(write_data), 40 .read_data(read_data), 42 . ram_clk(ram_clk), . ram_we_b(ram_we_b), 44 .ram_address(ram_address), .ram_data(ram_data), .ram_cen_b(ram_cen_b) 46 ); 48 mix delay_mix( //instantiate mix for wet/dry signal control 50 .reset (reset), .clock(clock), 52 .ready(ready), .mix_select(mix_select), 54 .signal1 (sample_in), .signal2(sample_out), 56 .mixed(mix_out) ); 58 assign write_addr = cur_addr + (buffer_size -1'b1); 60 parameter WAIT = 2'b00; parameter WRITE = 2'b01; 62 parameter READ = 2'b10; parameter COUNT = 2'b11; 64 66 reg [18:0] next_addr; reg [17:0] next_sample_out; reg [1:0] rcount , next_rcount; 68 always @( * ) begin //state machine for delay 70 case (rwstate) WAIT : begin 72 next_addr = cur_addr; //wait for ready, then go to WRITE 74 if (ready) begin next_rwstate = WRITE; ``` ``` 76 end else begin next_rwstate = WAIT; 78 \mathbf{end} \mathbf{end} 80 WRITE: begin 82 next_addr = cur_addr; //write current sample to high end of buffer, go to READ if (!ready) begin next_rwstate = READ; 84 next_rcount = 2'd3; end 86 else begin 88 next_rwstate = WRITE; end 90 end READ: begin //wait two cycles for read information\ to\ be\ retrieved\ ,\ then\ latch\ ,\ output\ ,\ and\ go\ back\ to\ WAIT 92 if (rcount > 0) begin next_addr = cur_addr; 94 next\_rcount = rcount - 1; next_rwstate = READ; 96 end else begin 98 next\_addr = cur\_addr+1; next_sample_out = read_data; next_rwstate = WAIT; 100 end 102 \quad \text{end} \quad default: begin next_rwstate = WAIT; 104 next_addr = cur_addr; 106 end endcase 108 end assign we = (rwstate = WRITE); 110 112 always @(posedge clock) begin //transition\ registers\ on\ posedge\ clock if (reset) begin ``` ``` 114 rwstate <= 0;\\ \operatorname{cur} \operatorname{-addr} \ <= \ 0\,; 116 \mathbf{end} else begin 118 rwstate <= next_rwstate; \mathtt{cur\_addr} \mathrel{<=} \mathtt{next\_addr} \,; 120 sample\_out <= \ next\_sample\_out \,; rcount <= next_rcount;</pre> 122 \quad \mathbf{end} \quad \mathbf{end} 124 endmodule ``` ## A.6 ZBT Suuport for Delay (delay\_zbt.v) ``` 2 // Ike's simple ZBT RAM driver for the MIT 6.111 labkit 4 // Data for writes can be presented and clocked in immediately; the actual // writing to RAM will happen two cycles later. // Read requests are processed immediately, but the read data is not available 8 // until two cycles after the intial request. 10 // A clock enable signal is provided; it enables the RAM clock when high. 12 module zbt_6111(clk, cen, we, addr, write_data, read_data, ram_clk , ram_we_b , ram_address , ram_data , ram_cen_b); 14 input clk; // system clock // clock enable for gating ZBT cycles 16 input cen; input we; // write enable (active HIGH) input [18:0] addr; // memory address 18 input [35:0] write_data; // data to write output [35:0] read_data; // data read from memory 20 ram_clk; // physical line to ram clock output output ram_we_b; // physical line to ram we_b output [18:0] ram_address; // physical line to ram address inout [35:0] ram_data; // physical line to ram data ram_cen_b; // physical line to ram clock enable output 26 // clock enable (should be synchronous and one cycle high at a time) wire ram_cen_b = ~cen; 28 30 // create delayed ram_we signal: note the delay is by two cycles! // ie we present the data to be written two cycles after we is raised // this means the bus is tri-stated two cycles after we is raised. reg [1:0] 34 we_delay; 36 always @(posedge clk) we_{delay} \le cen ? \{we_{delay}[0], we\} : we_{delay}; ``` ``` 38 //\ create\ two-stage\ pipeline\ for\ write\ data 40 reg [35:0] write_data_old1; reg [35:0] write_data_old2; 42 {\bf always} \ @({\bf posedge} \ {\tt clk}\,) 44 if (cen) \{ write\_data\_old2 \;,\;\; write\_data\_old1 \} <= \{ write\_data\_old1 \;,\;\; write\_data \}; 46 // wire to ZBT RAM signals 48 assign ram_we_b = we; 50 assign ram_clk = ~clk; // RAM is not happy with our data hold // times if its clk edges equal FPGA's /\!/ so we clock it on the falling edges 52 // and thus let data stabilize longer 54 assign ram_address = addr; 56 assign ram\_data = we\_delay[1] ? write\_data\_old2 : \{36\{1'bZ\}\}; assign read_data = ram_data; 58 endmodule // zbt_-6111 ``` ## A.7 FIR (filter.v) ``` //Filter Module for 6.111 2010 Final Project 3 //Marc Resnick // 7 module filter ( input wire clock, reset, ready, input wire signed [17:0] x, output reg signed [27:0] y); 11 reg [4:0] index, offset; 13 wire signed [9:0] coeff; reg signed [17:0] sample [30:0]; coeffs31 coeffs(.index(index), .coeff(coeff)); 15 //Instantiate coefficient LUT 17 always @(posedge clock) begin if (reset) begin 19 y <= 0; 21 offset \le 0; index \ll 0; 23 end else if (ready) begin 25 //On ready, increment offset, read new sample, reset accumulator offset <= offset + 1; 27 sample[offset] \le x; y <= 0; 29 index \ll 0; 31 else if (!ready && (index < 5'b11111)) begin y <= y + (coeff * sample[offset-index]); //perform convolution using accumulator 33 index \le index + 1; 35 end else begin 37 y \le y; ``` ``` index <= index; end 39 end 41 43 endmodule 45 47 // // Coefficients for a 31-tap low-pass FIR filter with Wn=.125 (eg, 3kHz for a 49 // 48kHz sample rate). Since we're doing integer arithmetic, we've scaled // the coefficients by 2**10 51 \mid // Matlab command: round(fir1(30,.125)*1024) // 53 55 module coeffs31( input wire [4:0] index, 57 output reg signed [9:0] coeff // tools will turn this into a 31x10 ROM 59 always @(index) 61 case (index) 5'd0: coeff = -10'sd1; 5'd1: coeff = -10'sd1; 63 5'd2: coeff = -10'sd3; 5'd3: coeff = -10'sd5; 65 5'd4: coeff = -10'sd6; 5'd5: coeff = -10'sd7; 67 5'd6: coeff = -10'sd5; 5'd7: coeff = 10'sd0; 69 5'd8: coeff = 10'sd10; 5'd9: coeff = 10'sd26; 71 5'd10: coeff = 10'sd46; 5'd11: coeff = 10'sd69; 73 5'd12: coeff = 10'sd91; 5'd13: coeff = 10'sd110; 75 5'd14: coeff = 10'sd123; ``` ``` 77 5'd15: coeff = 10'sd128; 5'd16: coeff = 10'sd123; 5'd17: coeff = 10'sd110; 79 5'd18: coeff = 10'sd91; 5'd19: coeff = 10'sd69; 81 5'd20: coeff = 10'sd46; 83 5'd21: coeff = 10'sd26; 5'd22: coeff = 10'sd10; 85 5'd23: coeff = 10'sd0; 5'd24: coeff = -10'sd5; 5'd25: coeff = -10'sd7; 87 5'd26: coeff = -10'sd6; 5'd27: coeff = -10'sd5; 89 5'd28: coeff = -10'sd3; 5'd29: coeff = -10'sd1; 91 5'd30: coeff = -10'sd1; default: coeff = 10 \text{'hXXX}; 93 endcase 95 endmodule ``` # A.8 Mix (mix.v) ``` 3 // Mix Module for 6.111 2010 Final Project Marc\ Resnick 5 // 7 module mix(input reset, clock, ready, input signed [17:0] mix_select, input signed [17:0] signal1, signal2, output wire signed [17:0] mixed); reg signed [17:0] 11 sigout_d, sigout; reg signed [17:0] l_mix_select; //latch mix-select on positive edge of clock 13 always @(posedge ready) l_mix_select <= mix_select;</pre> 15 wire signed [4:0] mix_mask; 17 assign mix_mask = (l_mix_select[17:13]); //mask the lower bits, get 5 bit mix selector assign mixed = sigout; signal1_weight, signal2_weight; 19 wire signed [5:0] 21 assign signall_weight = {1'b0,((mix_mask >>> 1) + 8)}; //calculate weights of both signals assign signal2-weight = \{1'b0,(15-((mix_mask >>> 1) + 8))\}; //based on mask 23 reg signed [31:0] sig_weight1 , sig_weight2; sig1_mult, sig2_mult; 25 reg signed [31:0] always @(posedge clock) begin 27 {\rm sig1\_mult} <= {\rm signal1*signal1\_weight}; \\ \hspace*{1.5cm} //multiply \ by \ weight \ (out \ of \ 16) 29 sig2_mult <= signal2*signal2_weight; //divide by 16 to get actual weight sig_weight1 \ll (sig_mult/16); // 31 sig_weight2 \ll (sig_mult/16); sigout_d <= sig_weight1[17:0] + sig_weight2[17:0]; //add two weighted signals sigout <= sigout_d; //pipeline calculation 33 end 35 endmodule ``` # A.9 Pan (pan.v) ``` // 3 // Pan // for the Field-Programmable Audio Effects Rack 5 // author: Drew Shapiro module pan(clk, reset, ready, pan_select, input_signal, left_signal_out, right_signal_out); input clk , reset , ready; input signed [17:0] pan_select, input_signal; output signed [17:0] left_signal_out, right_signal_out; 11 //5-bits yields -16 (full left) to 15 (rull right) pan 13 \mathbf{wire} \ \mathbf{signed} \ [\, 4:0 \, ] \quad \text{pan-mask} \, ; 15 reg signed [17:0] l_pan_select; assign pan_mask = l_pan_select[17:13]; 17 wire signed [5:0] left_index, right_index; left_weight , right_weight; 19 reg signed [5:0] assign left_index = \{1'b0, ((pan_mask >>> 1) + 8)\}; 21 assign right_index = \{1'b0, (15-((pan_mask >>> 1) + 8))\}; 23 reg signed [31:0] left_weighted, right_weighted, left_mult, right_mult; assign left_signal_out = left_weighted [17:0]; 25 assign right_signal_out = right_weighted [17:0]; 27 always @(posedge clk) begin if(ready) l_pan_select <= reset ? 0 : pan_select;</pre> 29 left_mult <= (input_signal*left_weight);</pre> right_mult <= (input_signal*right_weight); 31 left_weighted <= (left_mult/16); right_weighted <= (right_mult/16); 33 end 35 //equal-power gain lookup table always @ ( * ) begin 37 case(left_index) ``` ``` 0: begin left_weight <= 16; 39 right_weight <= 0; 41 end 1: begin left_weight <= 15; 43 right_weight <= 4; 45 end 2: begin 47 left_weight <= 15; right_weight <= 5; 49 end 3: begin 51 left_weight <= 14; right_weight <= 7; 53 end 4: begin 55 left_weight <= 14; right_weight <= 8; 57 end 5: begin left_weight <= 13; 59 right_weight <= 9; 61 end 6: begin left_weight <= 12; 63 right_weight <= 10; 65 \mathbf{end} 7: begin 67 left_-weight \ll 11; right_weight <= 11; 69 \mathbf{end} 8: begin 71 left_weight \ll 11; right_weight <= 11; 73 \mathbf{end} 9: begin left_weight <= 10; 75 right_weight <= 12; ``` ``` 77 \mathbf{end} 10: begin left_weight <= 9; 79 right_weight <= 13; 81 11: begin 83 left_weight <= 8; right_weight <= 14; 85 12: begin left_weight <= 7; 87 right_weight <= 14; 89 13: begin left_weight <= 5; 91 right_weight <= 15; 93 end 14: begin left_weight <= 4; 95 right_weight <= 15; 97 15: begin left_weight <= 0; right_weight <= 16; 101 endcase 103 \quad \text{end} \quad end module \\ ``` # A.10 Patchbay (patchbay.v) ``` 'timescale 1ns / 1ps 4 // Patchbay - provides physical interface for routing table update // for the Field-Programmable Audio Effects Rack 6 // // module patchbay(/*AUTOARG*/ 10 // Outputs scan\_channels\;,\; update\_routing\;,\; src\;,\; dest\;,\; scan\_hit\;,\; scan\_miss\;,\; match\_history\;, 12 // Inputs reset, clock, probe_channels 14 //TODO: read external routing table to silence redundant updates... parameter LOG.N = 4; localparam N = (1 \ll LOG_N); output [LOG_N-1:0] src , dest; 18 reset, clock; input 20 reg [2*LOG_N-1:0] src_dest; input [N-1:0] probe_channels; output [N-1:0] scan_channels; reg [N-1:0] scan_pattern; assign scan_channels = (1'b1 \ll (src)); 26 output update_routing; 28 update; reg output reg [N-1:0] match_history; 30 scan_hit, scan_miss; assign scan_hit = (probe_channels[dest] == 0); 32 assign src = src_dest[LOG_N-1:0]; //routing table address assign scan_miss = (src == {LOG_N{1'b1}})&& (& match_history); 34 assign dest = src_dest[2*LOG_N-1:LOG_N]; //routing table data 36 assign update_routing = scan_miss | scan_hit; ``` ``` 38 \mathbf{always} \ @ \ (\mathbf{posedge} \ \mathtt{clock}\,) \ \mathbf{begin} if(reset) begin \operatorname{src}_{-}\operatorname{dest} <= 0; 40 match\_history \le \{N\{1'b1\}\}; 42 \mathbf{end} else begin 44 src_dest \le src_dest + 1; match\_history <= \ \{\,match\_history\,[\,1\,4:0\,]\,\,,\,probe\_channels\,[\,dest\,]\,\}\,; 46 \mathbf{end} \mathbf{end} 48 endmodule ``` # A.11 Signal Generator (signal\_generator.v) ``` 1 'timescale 1ns / 1ps 3 // // Signal Generator Module for 6.111 2010 Final Project 5 // Marc Resnick/Drew Shapiro 7 module signal_generator(freq_select, clock, reset, ready, shape, wave_in, wave_out); parameter CLK_FREQ = 27\_000\_000; //27Mhz clock on lab kit input signed [17:0] freq_select; localparam SINE = 2'b00; 11 localparam SQUARE = 2'b01; localparam SAWTOOTH = 2'b10; 13 localparam TRIANGLE = 2'b11; 15 input clock , reset , ready; 17 input [1:0] shape; input signed [17:0] wave_in; output signed [17:0] wave_out; 19 21 //select the appropriate signal to expose reg signed [17:0] value; 23 //assign\ wave = value; 25 wire signed [17:0] mix_select; mix wet_dry_mix ( //allow signals to be chained/combined 29 .reset (reset), .clock(clock), 31 .ready(ready), .mix_select(18'b0), .signal1 (value), 33 .signal2(wave_in), .mixed(wave_out) 35 ); 37 ``` ``` reg [9:0] index; 39 reg [24:0] count, count_stop; always @(posedge clock) begin 41 if(reset) begin 43 index \ll 0; count <= 0; 45 count\_stop <= 0; end else begin 47 count \le (count = count\_stop) ? 0 : count + 1; //count until count_stop is reached count_stop <= $unsigned(freq_select) >> 10; 49 //then increment index and reset count index <= (count == count_stop) ? ((index == 10'd1023) ? 10'b0 : index + 1) : index; 51 //also reset index when it reaches 1023 53 end end 55 wire signed [17:0] square_value, sine_value, sawtooth_value, triangle_value; 57 //make 4 different shapes available assign square_value = (index > 10'd511)? 18'b10000000000000000 : 18'b0111111111111111111; //logic for square wave sinelut lut_sine(.THETA(index), .SINE(sine_value)); //and lookup tables for 61 sawtoothlut lut_sawtooth(.index(index), .wave(sawtooth_value)); //input is index , output is value trianglelut lut_triangle(.index(index), .wave(triangle_value)); //LUTs are full-wave, 1024 samples 63 65 always @ ( * ) //select wave to output case (shape) SINE: value = sine_value; 67 SQUARE: value = square_value; 69 SAWTOOTH: value = sawtooth_value; TRIANGLE: value = triangle_value; 71 endcase 73 endmodule ```