382 lines
19 KiB
Mathematica
382 lines
19 KiB
Mathematica
|
|
function [trackResults, channel]= tracking_V0_IQ(fid, channel, settings)
|
||
|
|
% Performs code and carrier tracking for all channels.
|
||
|
|
%
|
||
|
|
%[trackResults, channel] = tracking(fid, channel, settings)
|
||
|
|
%
|
||
|
|
% Inputs:
|
||
|
|
% fid - file identifier of the signal record.
|
||
|
|
% channel - PRN, carrier frequencies and code phases of all
|
||
|
|
% satellites to be tracked (prepared by preRum.m from
|
||
|
|
% acquisition results).
|
||
|
|
% settings - receiver settings.
|
||
|
|
% Outputs:
|
||
|
|
% trackResults - tracking results (structure array). Contains
|
||
|
|
% in-phase prompt outputs and absolute starting
|
||
|
|
% positions of spreading codes, together with other
|
||
|
|
% observation data from the tracking loops. All are
|
||
|
|
% saved every millisecond.
|
||
|
|
|
||
|
|
%--------------------------------------------------------------------------
|
||
|
|
% SoftGNSS v3.0
|
||
|
|
%
|
||
|
|
% Copyright (C) Dennis M. Akos
|
||
|
|
% Written by Darius Plausinaitis and Dennis M. Akos
|
||
|
|
% Based on code by DMAkos Oct-1999
|
||
|
|
%--------------------------------------------------------------------------
|
||
|
|
%This program is free software; you can redistribute it and/or
|
||
|
|
%modify it under the terms of the GNU General Public License
|
||
|
|
%as published by the Free Software Foundation; either version 2
|
||
|
|
%of the License, or (at your option) any later version.
|
||
|
|
%
|
||
|
|
%This program is distributed in the hope that it will be useful,
|
||
|
|
%but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
|
%MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
|
%GNU General Public License for more details.
|
||
|
|
%
|
||
|
|
%You should have received a copy of the GNU General Public License
|
||
|
|
%along with this program; if not, write to the Free Software
|
||
|
|
%Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
|
||
|
|
%USA.
|
||
|
|
%--------------------------------------------------------------------------
|
||
|
|
|
||
|
|
%CVS record:
|
||
|
|
%$Id: tracking.m,v 1.14.2.32 2007/01/30 09:45:12 dpl Exp $
|
||
|
|
|
||
|
|
%% Initialize result structure ============================================
|
||
|
|
|
||
|
|
% Channel status: trạng thái kênh
|
||
|
|
trackResults.status = '-'; % No tracked signal, or lost lock
|
||
|
|
|
||
|
|
% The absolute sample in the record of the C/A code start: khởi tạo mẫu tuyệt đối
|
||
|
|
% trong bản ghi C/A
|
||
|
|
trackResults.absoluteSample = zeros(1, settings.msToProcess);
|
||
|
|
|
||
|
|
% Freq of the C/A code: tần số mã C/A
|
||
|
|
trackResults.codeFreq = inf(1, settings.msToProcess);
|
||
|
|
|
||
|
|
% Frequency of the tracked carrier wave: tần số của sóng mang được theo dõi
|
||
|
|
trackResults.carrFreq = inf(1, settings.msToProcess);
|
||
|
|
|
||
|
|
% Outputs from the correlators (In-phase): đầu ra của các bộ tương quan (I)
|
||
|
|
trackResults.I_P = zeros(1, settings.msToProcess);
|
||
|
|
trackResults.I_E = zeros(1, settings.msToProcess);
|
||
|
|
trackResults.I_L = zeros(1, settings.msToProcess);
|
||
|
|
|
||
|
|
% Outputs from the correlators (Quadrature-phase): đầu ra của các bộ tương
|
||
|
|
% quan (Q)
|
||
|
|
trackResults.Q_E = zeros(1, settings.msToProcess);
|
||
|
|
trackResults.Q_P = zeros(1, settings.msToProcess);
|
||
|
|
trackResults.Q_L = zeros(1, settings.msToProcess);
|
||
|
|
|
||
|
|
% Loop discriminators
|
||
|
|
trackResults.dllDiscr = inf(1, settings.msToProcess);
|
||
|
|
trackResults.dllDiscrFilt = inf(1, settings.msToProcess);
|
||
|
|
trackResults.pllDiscr = inf(1, settings.msToProcess);
|
||
|
|
trackResults.pllDiscrFilt = inf(1, settings.msToProcess);
|
||
|
|
|
||
|
|
%--- Copy initial settings for all channels -------------------------------
|
||
|
|
trackResults = repmat(trackResults, 1, settings.numberOfChannels);
|
||
|
|
|
||
|
|
%% Initialize tracking variables ==========================================
|
||
|
|
|
||
|
|
codePeriods = settings.msToProcess; % For GPS one C/A code is one ms (với GPS thì 1 mã C/A là 1 ms)
|
||
|
|
|
||
|
|
%--- DLL variables --------------------------------------------------------
|
||
|
|
% Define early-late offset (in chips): Xác định độ lệch sớm-muộn (tính
|
||
|
|
% bằng chip)
|
||
|
|
earlyLateSpc = settings.dllCorrelatorSpacing;
|
||
|
|
|
||
|
|
% Summation interval: khoảng tính tổng
|
||
|
|
PDIcode = 0.001;
|
||
|
|
|
||
|
|
% Calculate filter coefficient values: Tính toán giá trị hệ số bộ lọc
|
||
|
|
[tau1code, tau2code] = calcLoopCoef(settings.dllNoiseBandwidth, ...
|
||
|
|
settings.dllDampingRatio, ...
|
||
|
|
1.0);
|
||
|
|
|
||
|
|
%--- PLL variables --------------------------------------------------------
|
||
|
|
% Summation interval
|
||
|
|
PDIcarr = 0.001;
|
||
|
|
|
||
|
|
% Calculate filter coefficient values
|
||
|
|
[tau1carr, tau2carr] = calcLoopCoef(settings.pllNoiseBandwidth, ...
|
||
|
|
settings.pllDampingRatio, ...
|
||
|
|
0.25);
|
||
|
|
% CARRIER TRACKING LOOP
|
||
|
|
Bl_carr =5; % Bandwith in hertz: băng thông tính bằng Hz
|
||
|
|
wn_carr = Bl_carr/0.7845; % Natural Frequency: tần số tự nhiên
|
||
|
|
% k_carr = 1; % Gain of the overall loop
|
||
|
|
% loop filter coefficients: hệ số lọc vòng lặp
|
||
|
|
a3=1.1*wn_carr^2;
|
||
|
|
b3=2.4*wn_carr;
|
||
|
|
wn3=wn_carr^3;
|
||
|
|
|
||
|
|
% filter values initialization: khởi tạo giá trị bộ lọc
|
||
|
|
olderrort_carr = 0;
|
||
|
|
oldolderrort_carr = 0;
|
||
|
|
oldcarrier_nco = 0;
|
||
|
|
oldoldcarrier_nco = 0;
|
||
|
|
T_int = 1e-3; % integration time
|
||
|
|
Acoeff=((T_int^2)*wn3/4)+(a3*T_int/2)+b3;
|
||
|
|
Bcoeff=((T_int^2)*wn3/2)-2*b3;
|
||
|
|
Ccoeff=((T_int^2)*wn3/4)-(a3*T_int/2)+b3;
|
||
|
|
|
||
|
|
hwb = waitbar(0,'Tracking...');
|
||
|
|
try
|
||
|
|
%% Start processing channels ==============================================
|
||
|
|
for channelNr = 1:settings.numberOfChannels
|
||
|
|
|
||
|
|
% Only process if PRN is non zero (acquisition was successful)
|
||
|
|
% Chỉ xử lý nếu PRN khác 0 (acquisition thành công)
|
||
|
|
if (channel(channelNr).PRN ~= 0)
|
||
|
|
% Save additional information - each channel's tracked PRN:
|
||
|
|
% Lưu thông tin bổ sung - PRN được tracked ở mỗi kênh
|
||
|
|
trackResults(channelNr).PRN = channel(channelNr).PRN;
|
||
|
|
|
||
|
|
% Move the starting point of processing. Can be used to start the
|
||
|
|
% signal processing at any point in the data record (e.g. for long
|
||
|
|
% records). In addition skip through that data file to start at the
|
||
|
|
% appropriate sample (corresponding to code phase). Assumes sample
|
||
|
|
% type is schar (or 1 byte per sample)
|
||
|
|
|
||
|
|
% Di chuyển điểm bắt đầu xử lý. Có thể được sử dụng để bắt đầu xử lý
|
||
|
|
% tín hiệu tại bất kỳ điểm nào trong bản ghi dữ liệu (ví dụ: đối với
|
||
|
|
% các bản ghi dài). Ngoài ra, hãy bỏ qua tệp dữ liệu đó để bắt đầu
|
||
|
|
% với mẫu thích hợp (tương ứng với giai đoạn mã). Giả sử loại mẫu là
|
||
|
|
% schar (hoặc 1 byte cho mỗi mẫu)
|
||
|
|
fseek(fid, ...
|
||
|
|
settings.skipNumberOfBytes + (channel(channelNr).codePhase-1)*settings.dataTypeSize*2, ...
|
||
|
|
'bof');
|
||
|
|
|
||
|
|
|
||
|
|
% Get a vector with the C/A code sampled 1x/chip
|
||
|
|
% Nhận 1 vector có mã C/A được lấy mẫu 1x/chip
|
||
|
|
caCode = generateCAcode(channel(channelNr).PRN);
|
||
|
|
% Then make it possible to do early and late versions: từ đó
|
||
|
|
% mới có thể tạo được early và late
|
||
|
|
caCode = [caCode(1023) caCode caCode(1)];
|
||
|
|
|
||
|
|
%--- Perform various initializations ------------------------------
|
||
|
|
% Thực hiện các khởi tạo khác nhau
|
||
|
|
|
||
|
|
% define initial code frequency basis of NCO
|
||
|
|
% Xác định tần số cơ sở mã ban đầu của NCO
|
||
|
|
codeFreq = settings.codeFreqBasis;
|
||
|
|
% define residual code phase (in chips)
|
||
|
|
% Xác định pha mã dư (trong chip)
|
||
|
|
remCodePhase = 0.0;
|
||
|
|
% define carrier frequency which is used over whole tracking period
|
||
|
|
% Xác định tần số sóng mang được sử dụng trong toàn bộ thời
|
||
|
|
% gian tracking
|
||
|
|
carrFreq = channel(channelNr).acquiredFreq;
|
||
|
|
carrFreqBasis = channel(channelNr).acquiredFreq;
|
||
|
|
% define residual carrier phase: xác định pha sóng mang dư
|
||
|
|
remCarrPhase = 0.0;
|
||
|
|
|
||
|
|
%code tracking loop parameters: tham số vòng lặp tracking
|
||
|
|
oldCodeNco = 0.0;
|
||
|
|
oldCodeError = 0.0;
|
||
|
|
|
||
|
|
%carrier/Costas loop parametersL: tham số vòng lặp sóng mang/
|
||
|
|
%Costas
|
||
|
|
oldCarrNco = 0.0;
|
||
|
|
oldCarrError = 0.0;
|
||
|
|
|
||
|
|
%=== Process the number of specified code periods =================
|
||
|
|
% Xử lý khoảng thời gian mã được chỉ định
|
||
|
|
for loopCnt = 1:codePeriods
|
||
|
|
%% GUI update -------------------------------------------------------------
|
||
|
|
% The GUI is updated every 50ms. This way Matlab GUI is still
|
||
|
|
% responsive enough. At the same time Matlab is not occupied
|
||
|
|
% all the time with GUI task.
|
||
|
|
|
||
|
|
% GUI được cập nhật cứ sau 50ms. Bằng cách này, GUI Matlab vẫn
|
||
|
|
% đủ đáp ứng. Đồng thời Matlab không phải lúc nào cũng bận rộn
|
||
|
|
% với nhiệm vụ GUI.
|
||
|
|
if (rem(loopCnt, 50) == 0)
|
||
|
|
try
|
||
|
|
waitbar(loopCnt/codePeriods, ...
|
||
|
|
hwb, ...
|
||
|
|
['Tracking: Ch ', int2str(channelNr), ...
|
||
|
|
' of ', int2str(settings.numberOfChannels), ...
|
||
|
|
'; PRN#', int2str(channel(channelNr).PRN), ...
|
||
|
|
'; Completed ',int2str(loopCnt), ...
|
||
|
|
' of ', int2str(codePeriods), ' msec']);
|
||
|
|
catch
|
||
|
|
% The progress bar was closed. It is used as a signal
|
||
|
|
% to stop, "cancel" processing. Exit.
|
||
|
|
disp('Progress bar closed, exiting...');
|
||
|
|
return
|
||
|
|
end
|
||
|
|
end
|
||
|
|
|
||
|
|
%% Read next block of data ------------------------------------------------
|
||
|
|
%% Đọc khối dữ liệu tiếp theo
|
||
|
|
% Find the size of a "block" or code period in whole samples
|
||
|
|
% Tìm kích thước của một "khối" hoặc đoạn mã trong toàn bộ mẫu
|
||
|
|
|
||
|
|
% Update the phasestep based on code freq (variable) and
|
||
|
|
% sampling frequency (fixed)
|
||
|
|
% Cập nhật bước pha dựa trên tần số mã (biến) và tần số lấy mẫu (cố định)
|
||
|
|
codePhaseStep = codeFreq / settings.samplingFreq;
|
||
|
|
|
||
|
|
blksize = ceil((settings.codeLength-remCodePhase) / codePhaseStep);
|
||
|
|
|
||
|
|
% Read in the appropriate number of samples to process this
|
||
|
|
% interation: Đọc số lượng mẫu thích hợp để xử lý sự tương tác này
|
||
|
|
[tmp, samplesRead] = fread(fid, ...
|
||
|
|
2*blksize, settings.dataType);
|
||
|
|
rawSignal=tmp(1:2:end)+1i*tmp(2:2:end);
|
||
|
|
|
||
|
|
rawSignal = transpose(rawSignal); %transpose vector: vector chuyển vị
|
||
|
|
|
||
|
|
% If did not read in enough samples, then could be out of
|
||
|
|
% data - better exit: Nếu không đọc đủ mẫu thì có thể hết
|
||
|
|
% dữ liệu - tốt hơn là thoát
|
||
|
|
if (samplesRead < 2*blksize)
|
||
|
|
disp('Not able to read the specified number of samples for tracking, exiting!')
|
||
|
|
fclose(fid);
|
||
|
|
return
|
||
|
|
end
|
||
|
|
|
||
|
|
%% Set up all the code phase tracking information -------------------------
|
||
|
|
%% Thiết lập thông tin theo dõi pha mã
|
||
|
|
% Define index into early code vector
|
||
|
|
tcode = (remCodePhase-earlyLateSpc) : ...
|
||
|
|
codePhaseStep : ...
|
||
|
|
((blksize-1)*codePhaseStep+remCodePhase-earlyLateSpc);
|
||
|
|
tcode2 = ceil(tcode) + 1;
|
||
|
|
earlyCode = caCode(tcode2);
|
||
|
|
|
||
|
|
% Define index into late code vector
|
||
|
|
tcode = (remCodePhase+earlyLateSpc) : ...
|
||
|
|
codePhaseStep : ...
|
||
|
|
((blksize-1)*codePhaseStep+remCodePhase+earlyLateSpc);
|
||
|
|
tcode2 = ceil(tcode) + 1;
|
||
|
|
lateCode = caCode(tcode2);
|
||
|
|
|
||
|
|
% Define index into prompt code vector
|
||
|
|
tcode = remCodePhase : ...
|
||
|
|
codePhaseStep : ...
|
||
|
|
((blksize-1)*codePhaseStep+remCodePhase);
|
||
|
|
tcode2 = ceil(tcode) + 1;
|
||
|
|
promptCode = caCode(tcode2);
|
||
|
|
|
||
|
|
remCodePhase = (tcode(blksize) + codePhaseStep) - 1023.0;
|
||
|
|
|
||
|
|
%% Generate the carrier frequency to mix the signal to baseband -----------
|
||
|
|
%% Tạo tần số sóng mang để trộn tín hiệu vào băng cơ sở
|
||
|
|
time = (0:blksize) ./ settings.samplingFreq;
|
||
|
|
|
||
|
|
% Get the argument to sin/cos functions
|
||
|
|
% Lấy đối số cho hàm sin/cos
|
||
|
|
trigarg = ((carrFreq * 2.0 * pi) .* time) + remCarrPhase;
|
||
|
|
remCarrPhase = rem(trigarg(blksize+1), (2 * pi));
|
||
|
|
|
||
|
|
% Finally compute the signal to mix the collected data to bandband
|
||
|
|
% Cuối cùng tính toán tín hiệu để trộn dữ liệu thu thập được vào băng thông
|
||
|
|
carrCos = cos(trigarg(1:blksize));
|
||
|
|
carrSin = sin(trigarg(1:blksize));
|
||
|
|
|
||
|
|
%% Generate the six standard accumulated values ---------------------------
|
||
|
|
%% Tạo sáu giá trị tích lũy tiêu chuẩn
|
||
|
|
% First mix to baseband
|
||
|
|
% Trộn đầu tiên vào baseband
|
||
|
|
iqBasebandSignal=(carrSin+1i*carrCos).* rawSignal;
|
||
|
|
|
||
|
|
|
||
|
|
|
||
|
|
qBasebandSignal = imag(iqBasebandSignal);
|
||
|
|
iBasebandSignal = real(iqBasebandSignal);
|
||
|
|
|
||
|
|
% Now get early, late, and prompt values for each
|
||
|
|
I_E = sum(earlyCode .* iBasebandSignal);
|
||
|
|
Q_E = sum(earlyCode .* qBasebandSignal);
|
||
|
|
I_P = sum(promptCode .* iBasebandSignal);
|
||
|
|
Q_P = sum(promptCode .* qBasebandSignal);
|
||
|
|
I_L = sum(lateCode .* iBasebandSignal);
|
||
|
|
Q_L = sum(lateCode .* qBasebandSignal);
|
||
|
|
|
||
|
|
%% Find PLL error and update carrier NCO ----------------------------------
|
||
|
|
|
||
|
|
% Implement carrier loop discriminator (phase detector)
|
||
|
|
% Triển khai bộ phân biệt vòng lặp sóng mang (bộ dò pha)
|
||
|
|
carrError = atan(Q_P / I_P) / (2.0 * pi);
|
||
|
|
|
||
|
|
% Implement carrier loop filter and generate NCO command
|
||
|
|
% Triển khai bộ lọc vòng lặp sóng mang và tạo lệnh NCO
|
||
|
|
carrNco = oldCarrNco + (tau2carr/tau1carr) * ...
|
||
|
|
(carrError - oldCarrError) + carrError * (PDIcarr/tau1carr);
|
||
|
|
oldCarrNco = carrNco;
|
||
|
|
oldCarrError = carrError;
|
||
|
|
|
||
|
|
% Modify carrier freq based on NCO command
|
||
|
|
% Sửa đổi tần số sóng mang dựa trên lệnh NCO
|
||
|
|
carrFreq = carrFreqBasis + carrNco;
|
||
|
|
|
||
|
|
trackResults(channelNr).carrFreq(loopCnt) = carrFreq;
|
||
|
|
%% Implement phase loop filter and generate NCO command (second order open loop transfer function F(Z))
|
||
|
|
% errort_carr=atan(Q_P / I_P) / (2.0 * pi);
|
||
|
|
% errort_carr_filt=(Acoeff*errort_carr + Bcoeff*olderrort_carr + Ccoeff*oldolderrort_carr);
|
||
|
|
% carrNco = 2*oldcarrier_nco - oldoldcarrier_nco + errort_carr_filt;
|
||
|
|
% carrFreq = carrFreqBasis + carrNco; %% NCO integrator (with the added pole of the integrator we complete the second order loop transfer function)
|
||
|
|
%
|
||
|
|
% oldolderrort_carr = olderrort_carr;
|
||
|
|
% olderrort_carr = errort_carr;
|
||
|
|
% oldoldcarrier_nco = oldcarrier_nco ;
|
||
|
|
% oldcarrier_nco = carrNco;
|
||
|
|
|
||
|
|
%% Find DLL error and update code NCO -------------------------------------
|
||
|
|
%% Tìm lỗi DLL và cập nhật mã NCO
|
||
|
|
codeError = (sqrt(I_E * I_E + Q_E * Q_E) - sqrt(I_L * I_L + Q_L * Q_L)) / ...
|
||
|
|
(sqrt(I_E * I_E + Q_E * Q_E) + sqrt(I_L * I_L + Q_L * Q_L));
|
||
|
|
|
||
|
|
% Implement code loop filter and generate NCO command
|
||
|
|
% Triển khai bộ lọc vòng lặp mã và tạo lệnh NCO
|
||
|
|
codeNco = oldCodeNco + (tau2code/tau1code) * ...
|
||
|
|
(codeError - oldCodeError) + codeError * (PDIcode/tau1code);
|
||
|
|
oldCodeNco = codeNco;
|
||
|
|
oldCodeError = codeError;
|
||
|
|
|
||
|
|
% Modify code freq based on NCO command
|
||
|
|
% Sửa đổi tần số mã dựa trên lệnh NCO
|
||
|
|
codeFreq = settings.codeFreqBasis - codeNco+carrFreq*codeFreq/(-carrFreq+1575.42e6);
|
||
|
|
|
||
|
|
trackResults(channelNr).codeFreq(loopCnt) = codeFreq;
|
||
|
|
|
||
|
|
%% Record various measures to show in postprocessing ----------------------
|
||
|
|
%% Ghi lại các biện pháp khác nhau để hiển thị trong quá trình xử lý hậu kỳ
|
||
|
|
% Record sample number (based on 8bit samples)
|
||
|
|
trackResults(channelNr).absoluteSample(loopCnt) = (1/settings.dataTypeSize/2)*ftell(fid);
|
||
|
|
trackResults(channelNr).remCodePhase(loopCnt) = remCodePhase;
|
||
|
|
|
||
|
|
|
||
|
|
trackResults(channelNr).dllDiscr(loopCnt) = codeError;
|
||
|
|
trackResults(channelNr).dllDiscrFilt(loopCnt) = codeNco;
|
||
|
|
trackResults(channelNr).pllDiscr(loopCnt) = carrError;
|
||
|
|
trackResults(channelNr).pllDiscrFilt(loopCnt) = carrNco;
|
||
|
|
|
||
|
|
trackResults(channelNr).I_E(loopCnt) = I_E;
|
||
|
|
trackResults(channelNr).I_P(loopCnt) = I_P;
|
||
|
|
trackResults(channelNr).I_L(loopCnt) = I_L;
|
||
|
|
trackResults(channelNr).Q_E(loopCnt) = Q_E;
|
||
|
|
trackResults(channelNr).Q_P(loopCnt) = Q_P;
|
||
|
|
trackResults(channelNr).Q_L(loopCnt) = Q_L;
|
||
|
|
end % for loopCnt
|
||
|
|
|
||
|
|
% If we got so far, this means that the tracking was successful
|
||
|
|
% Now we only copy status, but it can be update by a lock detector
|
||
|
|
% if implemented
|
||
|
|
% Nếu chúng tôi đã đi xa đến thế, điều này có nghĩa là việc theo dõi đã thành công
|
||
|
|
% Bây giờ chúng tôi chỉ sao chép trạng thái, nhưng nó có thể được cập nhật
|
||
|
|
% bằng trình phát hiện khóa nếu được triển khai
|
||
|
|
trackResults(channelNr).status = channel(channelNr).status;
|
||
|
|
|
||
|
|
end % if a PRN is assigned
|
||
|
|
end % for channelNr
|
||
|
|
catch exception
|
||
|
|
disp(exception.message);
|
||
|
|
end;
|
||
|
|
% Close the waitbar
|
||
|
|
close(hwb)
|