Phase-shift keying¶
import matplotlib.pyplot as plt
import numpy as np
import sdr
%config InlineBackend.print_figure_kwargs = {"facecolor" : "w"}
%matplotlib inline
# %matplotlib widget
In the sdr
library, phase-shift keying modulation is available in the sdr.PSK
class.
def analyze_psk(psk, esn0):
# Generate random decimal symbols
s = np.random.randint(0, psk.order, 100_000)
# Modulate decimal symbols to complex symbols
x = psk.map_symbols(s)
# Add AWGN to complex symbols to achieve desired Es/N0
snr = sdr.esn0_to_snr(esn0, sps=1)
x_hat = sdr.awgn(x, snr)
plt.figure(figsize=(10, 5))
plt.subplot(1, 2, 1)
sdr.plot.symbol_map(psk.symbol_map, limits=(-2, 2))
plt.subplot(1, 2, 2)
sdr.plot.constellation(x_hat, bins=75, heatmap=True, limits=(-2, 2))
plt.title(f"Constellation at {esn0} dB $E_s/N_0$")
plt.suptitle(f"{psk.order}-PSK constellation")
plt.tight_layout()
plt.show()
y = psk.modulate(s)
# h_srrc = sdr.root_raised_cosine(0.1, 6, sps)
# tx_mf = sdr.Interpolator(sps, h_srrc)
# y = tx_mf(x)
plt.figure(figsize=(10, 5))
sdr.plot.time_domain(y[0:1000])
plt.show()
Constellations¶
BPSK¶
bpsk = sdr.PSK(2, sps=10, pulse_shape="srrc")
analyze_psk(bpsk, 6)
data:image/s3,"s3://crabby-images/d71fc/d71fc2d67c53cf81266bebaa2b55914d83c151fe" alt="../../_images/44ddd676d80d87af4b45f2dfeebea16c2f4c0de764eecfbce37e99e0724a781f.png"
data:image/s3,"s3://crabby-images/658ec/658ec8802218dfa7538543c28878937da437ec09" alt="../../_images/a0181b3e746027c3a599d193175c081c874c1b70c49cf66c0bc4ae5d42099506.png"
QPSK¶
qpsk = sdr.PSK(4, phase_offset=45, sps=10, pulse_shape="srrc")
analyze_psk(qpsk, 9)
data:image/s3,"s3://crabby-images/d5584/d5584bc81ac3ed00e4ca8bc215c588a490e6aeef" alt="../../_images/eeafebaf9cc7ec6632a1fde3b0390a385d08ad7bfba9faf89ef09bb973a0f2dd.png"
data:image/s3,"s3://crabby-images/13e32/13e322297f8886dd9d202524eec5c94b9872c7c8" alt="../../_images/a76252c66d918ed8d76c1e51df5e75a019d33a8bf94714a5eca6d682011dc323.png"
8-PSK¶
psk8 = sdr.PSK(8, sps=10, pulse_shape="srrc")
analyze_psk(psk8, 12)
data:image/s3,"s3://crabby-images/17035/170353944e8697dddf8384b12e31133c6c88c406" alt="../../_images/b1540b98e9b8402ce88ca59c6fd04326ec7312f15142c836db4290ff591812be.png"
data:image/s3,"s3://crabby-images/28c16/28c16f61bc965e7becbb8a9184ece5448cd1a5a6" alt="../../_images/b87a649852693af588d2b61b7877489cd447886e2cec912cadc5ca1aa64f9590.png"
16-PSK¶
psk16 = sdr.PSK(16, sps=10, pulse_shape="srrc")
analyze_psk(psk16, 18)
data:image/s3,"s3://crabby-images/bc4a2/bc4a236624b784598ce46f7c1c752eb5a2f40ecb" alt="../../_images/e957629bbcf515f208fdb5d77a19ed896c81c233585868fe057a73b32ff83853.png"
data:image/s3,"s3://crabby-images/47c75/47c75591c260c031336b0e9b546918120e10b440" alt="../../_images/d36610d9d3fcec87479ccbe927530de05e5789dfc4e19c6be778e55b069299cd.png"
Error rate curves¶
def error_rates(psk, ebn0):
esn0 = sdr.ebn0_to_esn0(ebn0, psk.bps)
snr = sdr.esn0_to_snr(esn0)
ber = sdr.ErrorRate()
ser = sdr.ErrorRate()
for i in range(snr.size):
s = np.random.randint(0, psk.order, int(1e6))
a = psk.map_symbols(s)
a_tilde = sdr.awgn(a, snr[i])
s_hat, a_hat = psk.decide_symbols(a_tilde)
ber.add(ebn0[i], sdr.unpack(s, psk.bps), sdr.unpack(s_hat, psk.bps))
ser.add(esn0[i], s, s_hat)
return ber, ser
ebn0 = np.linspace(-2, 10, 20)
bpsk_ber, bpsk_ser = error_rates(bpsk, ebn0)
qpsk_ber, qpsk_ser = error_rates(qpsk, ebn0)
psk8_ber, psk8_ser = error_rates(psk8, ebn0)
psk16_ber, psk16_ser = error_rates(psk16, ebn0)
Bit error rate curves¶
plt.figure(figsize=(10, 5))
ebn0 = np.linspace(-2, 10, 200)
sdr.plot.ber(ebn0, bpsk.ber(ebn0), label="BPSK theoretical")
sdr.plot.ber(ebn0, qpsk.ber(ebn0), label="QPSK theoretical")
sdr.plot.ber(ebn0, psk8.ber(ebn0), label="8-PSK theoretical")
sdr.plot.ber(ebn0, psk16.ber(ebn0), label="16-PSK theoretical")
plt.gca().set_prop_cycle(None)
sdr.plot.ber(*bpsk_ber.error_rates(), linestyle="none", marker=".", label="BPSK simulated")
sdr.plot.ber(*qpsk_ber.error_rates(), linestyle="none", marker=".", label="QPSK simulated")
sdr.plot.ber(*psk8_ber.error_rates(), linestyle="none", marker=".", label="8-PSK simulated")
sdr.plot.ber(*psk16_ber.error_rates(), linestyle="none", marker=".", label="16-PSK simulated")
plt.ylim(1e-6, 1e0)
plt.title("Bit error rate curves for PSK modulation in AWGN")
plt.tight_layout()
plt.show()
data:image/s3,"s3://crabby-images/1fe35/1fe35ba33622918d90877d14df94b4e26557c4a5" alt="../../_images/30fcb824556beef452a166873d4ab790ae685b153ec34b7d6424ec0911e51ac5.png"
Symbol error rate curves¶
plt.figure(figsize=(10, 5))
esn0 = np.linspace(-4, 16, 200)
sdr.plot.ser(esn0, bpsk.ser(esn0), label="BPSK theoretical")
sdr.plot.ser(esn0, qpsk.ser(esn0), label="QPSK theoretical")
sdr.plot.ser(esn0, psk8.ser(esn0), label="8-PSK theoretical")
sdr.plot.ser(esn0, psk16.ser(esn0), label="16-PSK theoretical")
plt.gca().set_prop_cycle(None)
sdr.plot.ser(*bpsk_ser.error_rates(), linestyle="none", marker=".", label="BPSK simulated")
sdr.plot.ser(*qpsk_ser.error_rates(), linestyle="none", marker=".", label="QPSK simulated")
sdr.plot.ser(*psk8_ser.error_rates(), linestyle="none", marker=".", label="8-PSK simulated")
sdr.plot.ser(*psk16_ser.error_rates(), linestyle="none", marker=".", label="16-PSK simulated")
plt.ylim(1e-6, 1e0)
plt.title("Symbol error rate curves for PSK modulation in AWGN")
plt.tight_layout()
plt.show()
data:image/s3,"s3://crabby-images/7ff4f/7ff4fc6d09e0c1cd0c9b569d305ef0cabb5e0363" alt="../../_images/f56762034abd17fcb032735cb74f29d8738ec4d31997453841e5b4e43e529974.png"
Symbol mapping¶
The mapping of decimal symbols to complex symbols is important.
Ideally, adjacent symbol errors only result in 1 bit error.
This is generally accomplished using a Gray code (the default in :class:sdr.PSK
).
psk8_bin = sdr.PSK(8, symbol_labels="bin")
psk8_gray = sdr.PSK(8, symbol_labels="gray")
plt.figure(figsize=(10, 5))
plt.subplot(1, 2, 1)
sdr.plot.symbol_map(psk8_bin.symbol_map, limits=(-2, 2))
plt.title(f"Symbol map for binary coded 8-PSK")
plt.subplot(1, 2, 2)
sdr.plot.symbol_map(psk8_gray.symbol_map, limits=(-2, 2))
plt.title(f"Symbol map for gray coded 8-PSK")
plt.tight_layout()
plt.show()
data:image/s3,"s3://crabby-images/e4be9/e4be9bb4b4e2d96526e0834710065b712c3d49fa" alt="../../_images/a3c2b8d900462bceb7efa1b4608b1ae4f00c4f34d3aff540305f9a0af6905093.png"
Since adjacent symbol errors can lead to multiple bit errors, binary-coded 8-PSK has worse bit error performance.
plt.figure(figsize=(10, 5))
ebn0 = np.linspace(-2, 10, 200)
sdr.plot.ber(ebn0, sdr.PSK(8, symbol_labels="bin").ber(ebn0), label="8-PSK w/ binary code")
sdr.plot.ber(ebn0, sdr.PSK(8, symbol_labels="gray").ber(ebn0), label="8-PSK w/ gray code")
plt.title("Bit error rate curves for 8-PSK modulation in AWGN")
plt.tight_layout()
plt.show()
data:image/s3,"s3://crabby-images/0ed01/0ed0151fc99e527e183d71536c26b06b27580264" alt="../../_images/a5be79c9cc2ca103234c35416d9d2a12a0f89eb82000110001a1fff534093269.png"
Last update:
Dec 09, 2023