-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathlabcomdig.py
538 lines (446 loc) · 19.3 KB
/
labcomdig.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
'''
Conjunto de funciones desarrolladas para el manual de Laboratorio de Comunicaciones Digitales
impartido en la Escuela Técnica Superior de Ingeniería de la Universidad de Sevilla
'''
import numpy as np
'''
Tema 1
'''
def Qfunct(x):
'''
y = Qfunct(x) evalúa la función Q en x.
Donde y = 1/sqrt(2*pi) * integral desde x hasta inf de exp(-t^2/2) dt
'''
from scipy.special import erfc
from numpy import sqrt
y=(1/2)*erfc(x/sqrt(2))
return y
'''
Tema 2
'''
def LCDfmp(X):
'''
px,x = LCDfmp(X)
Estima la función de masa de probabilidad a partir de muestras de una
variable aleatoria discreta, en X.
Devuelve los valores de la fmp y los puntos en los que se evalua (soporte)
'''
Np = np.size(X)
x, px = np.unique(X, return_counts=True)
px = px/Np
return(px,x)
def LCDfdp(X, xpuntos=128):
'''
fxi,xi = LCDfdp(X, xpuntos=128)
Estima la función densidad de probabilidad a partir de muestras de una
variable aleatoria continua.
Recibe los valores de la variable aleatoria de la que queremos estimar
la función densidad de probabilidad y un parámetro opcional. Si no se da,
estima la fdp en 128 intervalos. En caso contrario, este parámetro puede
ser bien el número de intervalos o bien los extremos de los mismos.
Devuelve la estimación de la fdp en los puntos medios de los intervalos.
'''
fxi, xedge = np.histogram(X, bins=xpuntos, density=True)
xi=(xedge[1:]+xedge[:-1])/2
return fxi, xi
def fdpnormal(x,media=0,varianza=1):
'''
f = fdpnormal(x,media=0,varianza=1)
Calcula el valor de la fdp de una normal con media y varianza dadas en
los puntos dados por x. Si media y varianza no se dan, se ponen a 0 y
uno.
'''
f= 1/np.sqrt(2*np.pi*varianza)*np.exp(-(x-media)**2/(2*varianza));
return(f)
'''
Tema 4
'''
def transmisorpam(Bn,Eb,M,p,L):
'''
[Xn,Bn,An,phi,alfabetopam] = transmisorpam(Bn,Eb,M,p,L)
Entradas:
Bn = Secuencia de dígitos binarios
Eb = Energía media por bit transmitida en Julios
M = Número de síímbolos del código PAM
p = Pulso paso de baja o paso de banda
L = Número de puntos a utilizar en la representación de un símbolo
Devuelve:
Xn = la señal de información (discreta)
Bn = La secuencia de dígitos binarios realmente transmitidos
An = La secuencia de niveles de amplitud transmitidos
phi = Pulso básico real normalizado (energía unidad)
alfabetopam = Los niveles de amplitud asociados a cada símbolo
'''
#Comprobación de parámetros de entrada
p=p.squeeze()
if len(Bn)<1 or Eb<=0 or M<2 or p@p==0 or L<1:
raise Exception('Error: revise los parámetros de entrada')
# Se obtienen en primer lugar los niveles asociado a cada símbolo ¿Cuántos bits hay en cada símbolo?
k = int(np.ceil(np.log2(M)))
M = 2**(k) # Se Ajusta M a una potencia de dos
# El alfabeto [Ver la ecuación (4.21)]
alfabetopam = np.sqrt(3*Eb*np.log2(M)/(M**2-1))*(2*(np.arange(M))-M+1)
# Si Bn no tiene una longitud múltiplo de k, se completa con ceros
Nb = len(Bn) # Número de bits a transmitir, actualizado
Bn = Bn.squeeze().astype(int) #Comprobación de int y dimensiones
Bn = np.r_[Bn,np.zeros(int(k*np.ceil(Nb/k)-Nb)).astype(int)] #
Nb = len(Bn) # Número de bits a transmitir tras la corrección
Ns = Nb//k # Número de símbolos a transmitir
# La secuencia generada
if M>2:
An = alfabetopam[gray2de(np.reshape(Bn,[Ns,k]))]
else:
An = alfabetopam[Bn]
# Comprobación de las longitudes y otros datos del pulso suministrado para
# hacer que el número de muestras del mismo sea efectivamente L
Ls = len(p)
if Ls<L:
p = np.r_[p, np.zeros(L-Ls)]
elif Ls>L:
print('La duración del pulso se trunca a {} muestras'.format(str(L)))
p = p[:L] #Debe modificarse si se quiere un pulso de más de L muestras
# Se normaliza la energía del pulso para obtener la base del sistema
phi = p / np.sqrt(p@p)
# Obtención del tren de pulsos, Xn = np.kron(An,phi) ó
Xn = np.reshape(np.reshape(An,[Ns,1])*phi,[Ns*L,]) #Debe modificarse si se quiere un pulso de más de L muestras
return [Xn,Bn,An,phi,alfabetopam]
'''
Tema 5
'''
def transmisorpsk(Bn,Eb,M,p1,p2,L):
'''
[Xn,Bn,An,phi1,phi2,alfabeto] = transmisorpsk(Bn,Eb,M,p1,p2,L)
Entradas:
Bn = Secuencia de dígitos binarios
Eb = Energía media por bit transmitida en Julios
M = Número de símbolos del código PSK
p1 = Pulso real de la componente en fase
p2 = Pulso real de la componente en cuadratura
L = Número de puntos que se utilizará en la representación de un símbolo
Devuelve:
Xn = la señal de información discreta
Bn = La secuencia de dígitos binarios realmente transmitidos
An = La secuencia de símbolos complejos transmitidos
phi1 = Pulso básico real normalizado (energía unidad) de la componente en fase
phi2 = Pulso básico real normalizado (energía unidad) de la componente en cuadratura
alfabeto = El alfabeto utilizado asociado a cada símbolo transmitido
'''
#Comprobación de parámetros de entrada
p1 = p1.squeeze(); p2 = p2.squeeze()
if len(Bn)<1 or Eb<=0 or M<2 or p1@p1==0 or p2@p2==0 or L<1:
raise Exception('Error: revise los parámetros de entrada')
#Definiciones
eps = np.finfo(float).eps # por debajo de este valor se considera cero
# Comprobación de pulsos
Ls1 = len(p1)
Ls2 = len(p2)
if Ls1<L:
p1 = np.r_[p1, np.zeros(int(L-Ls1))]
elif Ls1>L:
p1 = p1[:L]
print('La duración del pulso se trunca a {} muestras'.format(str(L)))
if Ls2<L:
p2 = np.r_[p2, np.zeros(int(L-Ls2))]
elif Ls2>L:
p2 = p2[:L]
print('La duración del pulso se trunca a {} muestras'.format(str(L)))
# Se comprueba la ortogonalidad
if abs(p1@p2) >= 1e2*eps:
raise Exception('No es posible realizar la transmisión')
# Se normalizan las energías de los pulsos
phi1 = p1 / np.sqrt(p1@p1)
phi2 = p2 / np.sqrt(p2@p2)
# Se genera el pulso complejo [Ver la ecuación (5.25)]
phi = phi1 - 1j*phi2
# Obtención de los niveles asociados a cada símbolo: alfabeto
k = int(np.ceil(np.log2(M))) # Número de bits por símbolo
M = 2**(k) # Se ajusta M a una potencia de dos
# El alfabeto [Ver la ecuación (5.24)]
alfabeto = np.sqrt(Eb*k)*np.exp(1j*2*np.pi*np.arange(M)/M) #empieza en 0
# Si la longitud de Bn no es múltiplo de k, se completa con ceros
Nb = len(Bn)
Bn = Bn.squeeze().astype(int) #Comprobación de int y dimensiones
Bn = np.r_[Bn,np.zeros(int(k*np.ceil(Nb/k)-Nb)).astype(int)] #
# Número de bits y símbolos que vamos a transmitir
Nb = len(Bn) # Número de bits que vamos a transmitir tras la corrección
Ns = int(Nb/k)
# La secuencia generada
if M>2:
An = alfabeto[gray2de(np.reshape(Bn,[Ns,k]))]
else:
An = alfabeto[Bn]
# Obtención del tren de pulsos, Xn = np.real(np.kron(An, phi)) ó
Xn = np.real(np.reshape(np.reshape(An,[Ns,1])*phi,[Ns*L,]))
return [Xn, Bn, An, phi1, phi2, alfabeto]
def transmisorqam(Bn, Eb, M1, M2, p1, p2, L):
'''
[Xn,BnI,BnQ,AnI,AnQ,AI,AQ,phi1,phi2] = transmisorqam(Bn, Eb, M1, M2,p1,p2,L)
Entradas:
Bn = Secuencia de dígitos binarios
Eb = Energía media por bit transmitida en Julios
M1 = Nº de símbolos de la componente en fase
M2 = Nº de símbolos de la componente en cuadratura
p1 = Pulso real de la componente en fase
p2 = Pulso real de la componente en cuadratura
L = Nº de puntos que vamos a utilizar en la representación un símbolo
Devuelve:
Xn = la señal de información digital
BnI = Bits transmitidos por la componente en fase
BnQ = Bits transmitidos por la componente en cuadratura
AnI = Niveles de amplitud transmitidos por la componente en fase
AnQ = Niveles de amplitud transmitidos por la componente en cuadratura
AI = Niveles de amplitud usados en la componente en fase
AQ = Niveles de amplitud usados en la componente en cuadratura
phi1 = Pulso básico normalizado (energía unidad) de la componente en fase
phi2 = Pulso básico normalizado (energía unidad) de la componente en cuadratura
'''
#Comprobación de parámetros de entrada
p1 = p1.squeeze(); p2 = p2.squeeze()
if len(Bn)<1 or Eb<=0 or M1<2 or M2<2 or p1@p1==0 or p2@p2==0 or L<1:
raise Exception('Error: revise los parámetros de entrada')
#Definiciones
eps = np.finfo(float).eps # por debajo de este valor se considera cero
#Comprobación de los pulsos básicos
Ls1 = len(p1)
Ls2 = len(p2)
if Ls1<L:
p1 = np.r_[p1, np.zeros(int(L-Ls1))]
elif Ls1>L:
p1 = p1[:L]
print('La duración del pulso se trunca a {} muestras'.format(str(L)))
if Ls2<L:
p2 = np.r_[p2, np.zeros(int(L-Ls2))]
elif Ls1>L:
p2 = p2[:L]
print('La duración del pulso se trunca a {} muestras'.format(str(L)))
#Normalicemos las energías de los pulsos
phi1 = 1/np.sqrt(p1@p1)*p1
phi2 = 1/np.sqrt(p2@p2)*p2
#Comprobemos la ortogonalidad
if np.abs(phi1@phi2) >= 1e2*eps:
raise Exception('Bases no ortogonales, no es posible realizar la transmisión')
#Ajuste de los parámetros
k1 = int(np.ceil(np.log2(M1))) #Número de bits de la componente en fase
M1 = 2**(k1) #Valor de M1 tras la corrección
k2 = int(np.ceil(np.log2(M2))) #Número de bist de la componente en cuadratura M2 = 2**(k2) #Valor de M2 tras la corrección
k = k1 + k2 #Número de bits en cada símbolo QAM
Nb = len(Bn); Bn = Bn.squeeze().astype(int) #Comprobación de int y dimensiones
Bn = np.r_[Bn,np.zeros(int(k*np.ceil(Nb/k)-Nb)).astype(int)]
#Obtención de la mitad de la distancia mínima entre símbolos para Eb dada
A = np.sqrt(3*Eb*np.log2(M1*M2)/(M1**2+M2**2-2))
#El alfabeto con los niveles
AI = A * (2*np.arange(M1)-M1+1)
AQ = A * (2*np.arange(M2)-M2+1)
#Separación de la secuencia de bits en las secuencias de las componentes en fase y cuadratura
BnI,BnQ = split(Bn,M1,M2)
NbI = len(BnI)
NbQ = len(BnQ)
#Obtención de la secuencia de símbolos de las componentes en fase y cuadratura
if M1>2:
AnI = AI[gray2de(np.reshape(BnI,[int(NbI/k1),k1]))]
else:
AnI = AI[BnI]
if M2>2:
AnQ = AQ[gray2de(np.reshape(BnQ,[int(NbQ/k2),k2]))]
else:
AnQ = AQ[BnQ]
Ns = len(AnI)
#Las componentes en fase, cuadratura y total de la señal discreta a transmitir
XnI=np.reshape(np.reshape(AnI,[Ns,1])*phi1,[Ns*L,]) #ó XnI=np.kron(AnI, phi1);
XnQ=np.reshape(np.reshape(AnQ,[Ns,1])*phi2,[Ns*L,]) #ó XnQ=np.kron(AnQ, phi2);
Xn = XnI+XnQ;
return Xn, BnI, BnQ, AnI, AnQ, AI, AQ, phi1, phi2
'''
Tema 8
'''
def transmisorppm(Bn, Eb, p, L, M):
'''
[Xn,Nb,M,phi] = transmisorppm(Bn, Eb, p, L, M)
Genera una señal PPM de acuerdo con los parámetros suministrados.
Bn = Secuencia de dígitos binarios.
Eb = Energia media transmitida en Julios.
p = Un vector que representa el pulso básico de la señal PPM que se desea generar. El resto de los pulsos se obtienen obtiene desplazando el dado.
L = número de puntos que vamos a utilizar para transmitir el pulso básico dentro de un símbolo.
M = número de símbolos del código PPM.
Devuelve:
Xn = la señal de información digital (discreta).
Nb = número real de bits transmitidos después de la posible corrección.
M = número total de símbolos.
phi=una matriz en la que cada fila representa el pulso normalizado que se utiliza para transmitir cada símbolo; esto es, cada fila representa un elemento de la base del espacio de señales.
'''
#Comprobación de parámetros de entrada
if len(Bn)<1 or Eb<=0 or M<2 or p@p==0 or L<1:
raise Exception('Error: revise los parámetros de entrada')
# Ajustemos el valor de M a una potencia de 2
k=int(np.ceil(np.log2(M)))
M=2**k
Es=k*Eb # Energía del símbolo
# Ajustemos la longitud del pulso básico y su energía
Np=len(p)
if Np<L:
p=np.pad(p , L-Np, 'constant')
else:
p=p[:L]
p=(1/np.sqrt(np.sum(p*p)))*p # Normalicemos su energía
# Ajustemos los bits a transmitir a un número entero de símbolos: Si la secuencia Bn no tiene una longitud múltiplo de k, se completa con ceros
Bn=np.pad(Bn, [0,int(k*np.ceil(len(Bn)/k)-len(Bn))], 'constant')
Bn = Bn.squeeze().astype(int) #Comprobación de int y dimensiones
Nb=len(Bn) # Número de bits que vamos a transmitir
Ns=int(Nb/k) # Número de símbolos
# Obtengamos la secuencia de M valores diferentes correspondientes a los símbolos a transmitir
An = np.reshape(Bn,[Ns,k]) @ 2**np.flip(np.arange(k))
# Construyamos la matriz formada por los vectores de la base.
phi=np.zeros([M,L*M])
for kk in range(M):
phi[kk,kk*L:kk*L+L]=p
# Obtención del tren de pulsos conjunto
Xn = list(map(lambda Ai: phi[Ai,:], An))
Xn = np.sqrt(Es)*np.concatenate(Xn)
return [Xn,Bn,Nb,M,phi]
'''
Otras funciones de carácter general
'''
def gray2de(b):
'''
Convierte cada columna de la matriz formada por dígitos binarios b en un vector
fila de los valores decimales correspondientes, aplicando codificación de Gray.
'''
if not isinstance(b, (np.ndarray, list, tuple)): # se comprueba que no sea np.array, lista o tupla
raise Exception('Error: la entrada no es un array, lista o tupla')
b = np.array(b) # Esto es para que admita también listas y tuplas
c = np.zeros_like(b); c[:,0] = b[:,0]
for i in range(1,np.shape(b)[1]):
c[:,i] = np.logical_xor(c[:,i-1], b[:,i])
c = np.fliplr(c) # Convierte los bits menos significativos en los más significativos
#Comprueba un caso especial.
[n,m] = np.shape(c)
if np.min([m,n]) < 1:
d = []
return
d = c @ 2**np.arange(m)
return d
def de2gray(d,n):
"""
b = de2gray(d,n)
Convierte un número decimal, d, en un vector binario, b, de longitud n
"""
c = np.zeros([len(d),int(n)])
for i in range(int(n)): d, c[:,i] = np.divmod(d,2)
c = np.fliplr(c); b = np.zeros_like(c); b[:,0] = c[:,0]; aux = b[:,0]
for i in range(1,int(n)):
b[:,i] = np.logical_xor(aux, c[:,i])
aux = np.logical_xor(b[:,i], aux)
return np.reshape(b,[-1]).astype(np.int)
def split(Bn, M1, M2):
'''
Entradas
Bn = una secuencia de símbolos binarios
M1 = nº de símbolos de la componente en fase
M2 = nº de símbolos de la componente en cuadratura
Salida
Bn = una secuencia de símbolos binarios
M1 = nº de símbolos de la componente en fase
M2 = nº de símbolos de la componente en cuadratura
'''
k1 = int(np.log2(M1))
k2 = int(np.log2(M2))
k = k1 + k2
#Longitud de la secuencia
Nb = len(Bn)
#Una matriz con Ns=Nb/k filas formadas por los k bits
W = np.reshape(Bn,[int(Nb/k),k])
#Extrae la submatriz formada por los k1 primeros bits y pone una fila tras otra
BnI = np.reshape(W[:,:k1],[k1*int(Nb/k)])
#Extrae la submatriz formada por los k2 bits restantes y pone una fila tras otra
BnQ = np.reshape(W[:,k1:],[k2*int(Nb/k)])
return BnI, BnQ
def simbolobit(An,alfabeto):
'''
Bn = simbolobit(An, alfabeto)
An = secuencia de símbolos pertenecientes al alfabeto
alfabeto = tabla con los símbolos utilizados en la transmisión
Bn = una secuencia de bit, considerando que los símbolos se habían
generado siguiendo una codificación de Gray
'''
k = np.log2(len(alfabeto)) # bits por símbolo
if k>1:
distancia = abs(alfabeto[0]-alfabeto[1])
indices = np.round((An-alfabeto[0])/distancia)
Bn = np.reshape(de2gray(indices,k),[int(k*len(An))])
else:
Bn = ((An/max(alfabeto))+1)/2
return Bn
def detecta(r, alfabeto):
'''
An = detecta(r, alfabeto)
r = secuencia a la entrada del detector, con estimaciones de s_i
alfabeto = tabla con los niveles de amplitud/símbolos
Genera:
An = una secuencia de símbolos pertenecientes al alfabeto de acuerdo con
una regla de distancia euclidiana mínima (mínima distancia)
'''
# Longitud de la secuencia
N = len(r)
# Inicializa
An = np.zeros(N)
for i in range(N):
ind = np.argmin(abs(r[i]-alfabeto)) #ind = np.where(abs(r[i]-alfabeto) == np.amin(abs(r[i]-alfabeto)))
An[i] = alfabeto[ind]
return An
def detectaSBF(r,alfabeto):
'''
An = detectaSBF(r,alfabeto)
r = secuencia a la entrada del detector, con estimaciones de s_i
alfabeto = tabla con los niveles de amplitud/símbolos
Genera:
An = una secuencia de símbolos pertenecientes al alfabeto de acuerdo con
una regla de distancia euclidiana mínima (mínima distancia)
'''
# Obtiene el índice respecto al alfabeto
ind = np.argmin(np.abs(np.repeat(np.expand_dims(r,1),len(alfabeto),1) - alfabeto), 1)
# Genera la secuencia de niveles detectados
An = alfabeto[list(ind)]
return An
# def split(Bn, M1, M2):
# '''
# Bn = una secuencia de símbolos binarios
# M1 = nº de símbolos de la componente en fase
# M2 = nº de símbolos de la componente en cuadratura
# Devuelve:
# BnI = La secuencia de símbolos binarios de la componente en fase
# BnQ = La secuencia de símbolos binarios de la componente en cuadratura
# '''
# k1=int(np.log2(M1))
# k2=int(np.log2(M2))
# k=k1+k2
# #Longitud de la secuencia
# Nb=len(Bn)
# #Una matriz con Nb/k filas formadas por los k1 bits más los k2 bits
# W=np.reshape(Bn,[int(Nb/k),k])
# #Extrae la submatriz formada por los k1 primeros bits y pone una fila tras otra
# BnI=np.reshape(W[:,:k1],[k1*int(Nb/k)])
# #Extrae la submatriz formada por los k2 bits restantes y pone una fila tras otra
# BnQ=np.reshape(W[:,k1:],[k2*int(Nb/k)])
# return BnI, BnQ
def une(BndetectadoI,BndetectadoQ,M1,M2):
'''
[Bndetectado]=une(BndetectadoI,BndetectadoQ,M1,M2)
BndetectadoI = una secuencia de símbolos binarios correspondientes a los bits en posiciones múltiplos de k1
BndetectadoQ = una secuencia de símbolos binarios correspondientes a los bits en posiciones múltiplos de k2
M1 = nº de símbolos de la componente en fase
M2 = nº de símbolos de la componente en cuadratura
'''
# Devuelve
# Bndetectado = Los bits entremezclados
import numpy as np
k1 = np.log2(M1)
k2 = np.log2(M2)
N = len(BndetectadoI)/k1
C1 = np.reshape(BndetectadoI,[int(N),int(k1)])
C2 = np.reshape(BndetectadoQ,[int(N),int(k2)])
C = np.c_[C1,C2]
Bndetectado = np.reshape(C,[int(N*(k1+k2))])
return Bndetectado