-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsimulateFrame.m
113 lines (93 loc) · 3.97 KB
/
simulateFrame.m
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
function [errorCount, errorRatio, timeErr, frequencyErr, channelErr] = simulateFrame(SNRdB)
% Simulation of a frame consisting of 68 OFDM symbols
% for debugging
simulateOffset = true;
simulateChannelEstimation = true;
%% Generate Data
nSamples = 6817;
kBits = 4; % Bits per sample
nOFDMsymbols = 68; % per frame
% Generate vector of random binary data.
dataIn = randi([0 1], 68, nSamples*kBits);
%% 16-QAM
dataMod = zeros(nOFDMsymbols, nSamples);
for i=1:nOFDMsymbols
dataMod(i,:) = modules.qamModulation(dataIn(i,:));
end
%% Pilot insertion
[dataModWithPilots, pilots, pilotPositions] = modules.pilotInsertion(dataMod);
%% OFDM modulator
ofdmSignalTX = modules.ofdmModulator(dataModWithPilots);
%% Channel simulation
SNRlin = 10^(SNRdB/10);
% transmit data over channel
channelLength = length(modules.channelGenerator); % 1370
channel = zeros(68,channelLength);
% ofdmSignalRX1 corresponds to a received frame of OFDM symbols
ofdmSignalRX1 = zeros(68, 10240);% + channelLength - 1);
for i=1:nOFDMsymbols
signalTX = ofdmSignalTX(i,:);
signalPower = sum(abs(signalTX).^2) / length(signalTX);
noisePower = signalPower / SNRlin;
% convolution with channel impulse response
channel(i,:) = modules.channelGenerator();
RXdataNoNoise = conv(signalTX,channel(i,:));
n = sqrt(noisePower/2) * (randn(1,length(RXdataNoNoise)) + 1j*randn(1,length(RXdataNoNoise)));
RXdata1 = RXdataNoNoise + n;
ofdmSignalRX1(i,:) = RXdata1(1:10240);
end
% reshape received OFDM frame to a row vector
ofdmSignalRX2 = reshape(ofdmSignalRX1',1,[]);
% time and frequency offset
timeOffset = randi([0,600],1);
frequencyOffsetMin = -1/2;
frequencyOffsetMax = 1/2;
frequencyOffset = (frequencyOffsetMax - frequencyOffsetMin) * rand() + frequencyOffsetMin;
ofdmSignalRXdelayed = [zeros(1,timeOffset), ofdmSignalRX2];
ofdmSignalRXdelayed(1:timeOffset) = sqrt(noisePower/2) * (randn(1,timeOffset) + 1j*randn(1,timeOffset));
m = 0:1:length(ofdmSignalRXdelayed)-1;
ofdmSignalRX = ofdmSignalRXdelayed .* exp(1i*2*pi*frequencyOffset*m/8192);
%% Synchronisation
if simulateOffset
[ofdmSignalRXsynchronized, timeOffsetEst, frequencyOffsetEst] = modules.offsetEstimator(ofdmSignalRX, SNRlin);
timeErr = (timeOffset - timeOffsetEst)^2;
frequencyErr = (frequencyOffset - frequencyOffsetEst)^2;
else
timeErr = 0;
frequencyErr = 0;
ofdmSignalRXsynchronized = ofdmSignalRX1; % without offset & synchronization
end
%% Demodulation
dataRX = modules.ofdmDemodulator(ofdmSignalRXsynchronized);
% extract data of used subcarriers
dataRXestimated = dataRX(:,1:6817);
%% Channel Estimation
Hest = modules.channelEstimation(dataRXestimated, pilots);
% assume perfect channel knowledge
H = zeros(nOFDMsymbols,8192);
for i=1:nOFDMsymbols
H(i,:) = fft(channel(i,:),8192);
end
H = H(:,1:6817);
% compute channel estimation error
channelErr = sum(abs(H(:)-Hest(:)).^2) / numel(H);
if simulateChannelEstimation
dataRXestimated = dataRXestimated ./ Hest;
else
dataRXestimated = dataRXestimated ./ H;
end
%% Demapping
dataDemappedEstimated = modules.symbolDemapping(dataRXestimated);
% Remove pilots and get serial stream of data
dataDemappedEstimated = dataDemappedEstimated.';
dataDemappedEstimated(logical(pilotPositions.')) = [];
dataOut = de2bi(dataDemappedEstimated, 'left-msb');
%% Compute number of bit errors
% Get original data stream
dataCompare = modules.symbolDemapping(dataMod);
dataCompare = dataCompare.';
dataCompare(logical(pilotPositions.')) = [];
dataCompare = de2bi(dataCompare.', 'left-msb');
% Compute errors
errorCount = sum(reshape(dataOut',1,[]) ~= reshape(dataCompare',1,[]));
errorRatio = errorCount / numel(dataOut);