Skip to content
Jon Dawson edited this page Jul 18, 2023 · 19 revisions

Pi Pico Rx

PCB]

Pi Pico Rx is a minimal SDR reciever based around the Raspbery Pi Pico. The design uses a "Tayloe" Quadrature Sampling Detector (QSD) popularised by Dan Tayloe. And used in many HF SDR radio designs. This simple, design allows a high quality mixer to be implemented using an inexpensive analogue switch.

A quadradure oscillator is generated using the PIO feature of the RP2040. This eliminaes the need to use an external programable oscillator. Without overclocking the device this supports frequencies up to about 30MHz, conveniently covering the LW, MW and SW bands.

The IQ output from the QSD is amplified using a high-speed, low-noise op-amp. The I and Q channels are sampled by the built-in ADC which provides 250kHz of bandwidth. The dual core ARM Cortex M0 processor implements the Digital Signal Processessing algorithms, demodulating AM, FM, SSB and CW to produce an audio output. worried that Audio output is provided using a PWM followed by a low-pass filter. At first I used an LM386 audio amplifier, but later found that with a suitable current limiting resistor the IO pin could easily drive a pair of headphones or even a small speaker directly.

A cobbled together prototype proves that it is possible to build an HF SDR reciever using an pi pico, an analogue switch and a handfull of discrete components.

VID_20230119_175559892.mp4

Sampling IQ data using a round-robin ADC

One of the challenges of using the internal ADC to sample the IQ signals is that the ADC is only capable of sampling one channel at a time. It can be configured in a round-robin mode that samples I and Q alternately. I had worried that this might create a phase imbalance between the I and Q channels. I needn't have worried, it turns out that there is a simple way to recover a complex signal with 250kHz bandwidth by sampling I and Q alternately at 500kSample/s.

The trick is to low pass filter the I and Q data leaving 250kHz of bandwidth from -125kHz to 125kHz. The QSD detector itself forms a low-pass filter, so this can easily be achieved by selecting suitable capacitor values in the op-amp. The ADC is configured to sample I and Q alternately (starting with I). In the software, the "missing" values can be replaced by zeros.

i = i0, 0, i2, 0, i4 ... q = 0, q1, 0, q3, 0 ...

This results in a complex signal with a sampling rate of 500KS/s. The central half of the spectrum from -125kHz to 125kHz contains the spectrum we want. The other half of the spectrum from -250kHz to -125kHz and 125kHz to 250kHz contain a reflection of the central half. To recover the original spectrum, we simply need to low-pass filter the signal so that we retain the central section. We could then reduce the sample rate to 250kHz.

Why does this work?

To understand why this works, it helps to think about how we could have converted the signal into a real (rather than complex IQ) signal, and sampled it using a single channel ADC. This is one of the approaches I had originaly considered taking.

To keep Nyquist happy, we need to filter the complex data so that all our signals sit between -125kHz and 125kHz. We could have then shifted the data up by 125kHz so that our signals are between 0 and 250kHz. The frequency shift is 1/4 of the 500KSample/second sample rate. A frequency shift by Fs/4 can be implemented by rotating the signal by 1/4 turn each sample. This doesn't need any multiplication, only negation.

i = i0, -q1, -i2, q3 q = q0, i1, -q2, -i3

Since our signal now only contains positive frequencies, the imaginary (Q) part of the signal doesn't contain any information and we can throw it away. A signal contining only real (I) parts has a symetrical spectrum, so we now have negative reflections of the positive signals.

i = i0, -q1, -i2, q3

We now have a real signal that could be sampled with a single channel ADC at 500kSamples/s. This could have implemented in hardware using a simple mixer, but we only need I and Q samples alternately, so we could use a round-robin ADC to sample I and Q alternatley, and implement the mixer in software negating I and Q when necassary.

Once we have the real signal in software, we might like to convert the real signal back to a complex one. We could use a Hilbert transform, this would filter out the negative frequencies leaving an asymetrical, complex signal containing only positive frequencies from 0 to 250kHz.

Another approach would be to shift the frequencies down by 125kHz leaving the original spectrum from -125kHz to 125kHz, now with reflections in the outer half of the spectrum. These could be removed with a low-pass filter. We can take the same approach to the Fs/4 frequency shift, this time rotating 1/4 turn each sample in the opposite direction.

i = i0, 0, i2, 0, i4 ... q = 0, q1, 0, q3, 0 ...

Inspecting the resulting samples, we can see that the downwards frequency shift has cancelled out the negations we performed during the upwards frequency shift, leaving us with the alternating I/Q samples we originaly captured.

Conveniently, it turns out, the alternating IQ samples captured from the round-robin ADC were the only samples we actually needed to fully capture the central half of the frequency spectrum. The "misssing" samples only contributed to the outer part of the spectrum that we had already filtered out.

Creating Quadrature Oscillator Using PIO

The pi pico is based on the RP2040 microcontroller. The PIO is a novel feature of the RP2040. Programable State Machines (like small microprocessors) can be configured to offload IO functions from the software. It is fairly simple to configure a PIO state machine to output a quadrature oscillator on 2 IO pins. Once configured the Oscillator runs autonomously without software intervention, not placing any further load on the CPU.

The PIO program is remarkably simple:

..code::

.program nco
set pins, 0
set pins, 1      ; Drive pin low
set pins, 3      ; Drive pin high
set pins, 2      ; Drive pin low

The freuquency of the NCO can be programmed using the PIO clock divider. This has a 16 bit integer and 256 bit fractional part. With an input clock of 125MHz, the NCO can be programmed from a few hundred Hz to just over 30MHz. Perfect for a LW/MW/SW reciever. At low frequency a good resolution can be achieved, but at high frequencies the step size can be more then 100kHz. However, with a bandwidth of 250kHz, that is still enough to give continuous coverage of the whole frequency range. To compensate for the course frequency adjustment in the hardware, a high resolution frequency shifter is implemented in software. (The 32 bit phase accumulator has a theoretical resolution a little over 0.0001 Hz which should be ample.)

Clone this wiki locally