mirror of
https://github.com/jhshi/openofdm.git
synced 2025-01-28 14:53:49 +00:00
123 lines
5.0 KiB
ReStructuredText
123 lines
5.0 KiB
ReStructuredText
Packet Detection
|
|
================
|
|
|
|
802.11 OFDM packets start with a short PLCP Preamble sequence to help the
|
|
receiver detect the beginning of the packet. The short preamble duration is
|
|
8 us. At 20 MSPS sampling rate, it contains 10 repeating sequence of 16 I/Q
|
|
samples, or 160 samples in total. The short preamble also helps the receiver
|
|
for coarse frequency offset correction , which will be discussed separately in
|
|
:ref:`freq_offset`.
|
|
|
|
|
|
Power Trigger
|
|
-------------
|
|
|
|
- **Module**: :file:`power_trigger.v`
|
|
- **Input**: ``sample_in`` (16B I + 16B Q), ``sample_in_strobe`` (1B)
|
|
- **Output**: ``trigger`` (1B)
|
|
- **Setting Registers**: ``SR_POWER_THRES``, ``SR_POWER_WINDOW``,
|
|
``SR_SKIP_SAMPLE``.
|
|
|
|
The core idea of detecting the short preamble is to utilize its repeating nature
|
|
by calculating the auto correlation metric. But before that, we need to make sure
|
|
we are trying to detect short preamble from "meaningful" signals. One example of
|
|
"un-meaningful" signal is constant power levels, whose auto correlation metric
|
|
is also very high (nearly 1) but obviously does not represent packet beginning.
|
|
|
|
The first module in the pipeline is the :file:`power_trigger.v`. It takes the I/Q
|
|
samples as input and asserts the ``trigger`` signal during a potential packet
|
|
activity. Optionally, it can be configured to skip the first certain number of
|
|
samples before detecting a power trigger. This is useful to skip the spurious
|
|
signals during the intimal hardware stabilization phase.
|
|
|
|
The logic of the ``power_trigger`` module is quite simple: after skipping
|
|
certain number of initial samples, it waits for significant power increase and
|
|
triggers the ``trigger`` signal upon detection. The ``trigger`` signal is
|
|
asserted until the power level is smaller than a threshold for certain number of
|
|
continuous samples.
|
|
|
|
Short Preamble Detection
|
|
------------------------
|
|
|
|
- **Module**: :file:`sync_short.v`
|
|
- **Input**: ``sample_in`` (16B I + 16B Q), ``sample_in_strobe`` (1B)
|
|
- **Output**: ``short_preamble_detected`` (1B)
|
|
- **Setting Registers**: ``SR_MIN_PLATEAU``
|
|
|
|
|
|
.. _fig_short_preamble:
|
|
.. figure:: /images/short_preamble.png
|
|
:align: center
|
|
|
|
In-Phase of Short Preamble.
|
|
|
|
:numref:`fig_short_preamble` shows the in-phase of the beginning of a packet.
|
|
Some repeating patterns can clearly be seen. We can utilize this characteristic
|
|
and calculate the auto correlation metric of incoming signals to detect such
|
|
pattern:
|
|
|
|
.. math::
|
|
|
|
corr[i] = \frac{\left\lVert\sum_{i=0}^{N}{S[i]*\overline{S[i+16]}}\right\rVert}
|
|
{\sum_{i=0}^{N}{S[i]*\overline{S[i]}}}
|
|
|
|
where :math:`S[i]` is the :math:`\langle I,Q \rangle` sample expressed as a
|
|
complex number, and :math:`\overline{S[i]}` is its conjugate, :math:`N` is the
|
|
correlation window size. The correlation
|
|
reaches 1 if the incoming signal is repeating itself every 16 samples. If the
|
|
correlation stays high for certain number of continuous samples, then a short
|
|
preamble can be declared.
|
|
|
|
.. _fig_corr:
|
|
.. figure:: /images/corr.png
|
|
:align: center
|
|
|
|
Auto Correlation of the Short Preamble samples (N=48).
|
|
|
|
To plot :numref:`fig_corr`, load the samples (see :ref:`sec_sample`), then:
|
|
|
|
.. code-block:: python
|
|
|
|
from matplotlib import pyplot as plt
|
|
|
|
fig, ax = plt.subplots(nrows=2, ncols=1, sharex=True)
|
|
ax[0].plot([s.real for s in samples[:500]], '-bo')
|
|
ax[1].plot([abs(sum([samples[i+j]*samples[i+j+16].conjugate()
|
|
for j in range(0, 48)]))/
|
|
sum([abs(samples[i+j])**2 for j in range(0, 48)])
|
|
for i in range(0, 500)], '-ro')
|
|
plt.show()
|
|
|
|
|
|
:numref:`fig_corr` shows the auto correlation value of the samples in
|
|
:numref:`fig_short_preamble`. We can see that the correlation value is almost 1
|
|
during the short preamble period, but drops quickly after that. We can also see
|
|
that for the very first 20 samples or so, the correlation value is also very
|
|
high. This is because the silence also repeats itself (at arbitrary interval)!
|
|
That's why we first use the ``power_trigger`` module to detect actual packet
|
|
activity and only perform short preamble detection on non-silent samples.
|
|
|
|
A straight forward implementation would require
|
|
both multiplication and division. However, on FPGAs devision consumes a lot of
|
|
resources so we really want to avoid it. In current implementation, we use a
|
|
fixed threshold (0.75) for the correlation so that we can use bit-shift to
|
|
achieve the purpose. In particular, we calculate ``numerator>>1 + numerator>>2``
|
|
and compare that with the denominator. For the correlation window size, we set
|
|
:math:`N=16`.
|
|
|
|
|
|
.. _fig_sync_short:
|
|
|
|
.. figure:: /images/sync_short.png
|
|
:align: center
|
|
|
|
``sync_short`` Module Diagram
|
|
|
|
:numref:`fig_sync_short` shows the internal module diagram of the ``sync_short``
|
|
module. In addition to the number of consecutive samples with correlation
|
|
larger than 0.75, the ``sync_short`` module also checks if the incoming signal
|
|
has both positive (> 25%) and negative (> 25%) samples to further eliminate
|
|
false positives (e.g., when the incoming signals are constant non-zero values).
|
|
Again, the thresholds (25%) are chosen so that we can use only bit-shifts for
|
|
the calculation.
|