
function [A,B,C,D,R,Q,GaA,GaB,GaC,GaD,mu,error]=Our_Method(Y,U,n,mm,p,T,Iter_Max,xi,rr,ts)

% Y: mm*T
% U: p*T
% R and Q are diagonal
% 
%% initial conditions

mu(:,:,1)=xi; 

if size(rr,1)==1
    Sig(:,:,1)=(rr^2)*eye(n);
else
    Sig(:,:,1)=rr*rr';
end



A=ts*eye(n); B=ts*eye(n,p); 
C=ts*eye(mm,n); D=ts*eye(mm,p); 

R=ts*eye(n);Q=ts*eye(mm);


p1=1e5;

GaA=p1*ones(n,n); GaB=p1*ones(n,p); GaC=p1*ones(mm,n); GaD=p1*ones(mm,p); 

a0=1e-5;b0=1e-5;

%% Iteration loop

error(:,1)=ones(6,1);threshold=1e-18;ncount=0; % thr: The threshold to stop the iteration  ncount: record the number of iteration3
while max(error(:,ncount+1))>threshold
    
    A2=A; % In order to compute || A^(k+1)-A^(k)||
    B2=B;C2=C;D2=D;R2=R;Q2=Q;
    
    ncount=ncount+1; %

    if ncount>Iter_Max
        break;
    end
        
    if mod(ncount,100)==0 
        fprintf('This is the %d-th iteration \n',ncount) 
    end

   %%  KF filter

   for i=1:T
       mubar(:,:,i)=A*mu(:,:,i)+B*U(:,i);
       Sigbar(:,:,i)=A*Sig(:,:,i)*A'+R;
       K(:,:,i)=Sigbar(:,:,i)*C'/(C*Sigbar(:,:,i)*C'+Q);
       mu(:,:,i+1)= mubar(:,:,i)+K(:,:,i)*(Y(:,i)-C*mubar(:,:,i)-D*U(:,i));
       Sig(:,:,i+1)=(eye(n)-K(:,:,i)*C)*Sigbar(:,:,i);
   end
   


  %%  Kalman smoother
  
  m(:,:,T+1)=mu(:,:,T+1);
  P(:,:,T+1)=Sig(:,:,T+1); 
   

   for i=1:T
       ii=T+1-i;
       G(:,:,ii)=Sig(:,:,ii)*A'/Sigbar(:,:,ii);  
       m(:,:,ii)=mu(:,:,ii)+G(:,:,ii)*(m(:,:,ii+1)-mubar(:,:,ii));% the mean of KS
       P(:,:,ii)=Sig(:,:,ii)+G(:,:,ii)*(P(:,:,ii+1)-Sigbar(:,:,ii))*G(:,:,ii)';% the variance of KS
   end
    
mu(:,:,1)=m(:,:,1);
Sig(:,:,1)=P(:,:,1);
 
   %% PC(:,:,i) is the covariance of X(i) and X(i-1)
   
   PC(:,:,T)=(eye(n)-K(:,:,T)*C)*A*Sig(:,:,T); % initial conditions
   
   for i=1:T-1
       ii=T-i;
       PC(:,:,ii)=Sig(:,:,ii+1)*G(:,:,ii)'+G(:,:,ii+1)*(PC(:,:,ii+1)-A*Sig(:,:,ii+1))*G(:,:,ii)';% 计算X(i)和X(i-1)之间的方差
   end

   %% update A
   for r=1:n

   MA1=zeros(1,n);MA2=zeros(n,n);

   for i=1:T
       MA1=MA1+(m(r,:,i+1)-B(r,:)*U(:,i))*m(:,:,i)'+PC(r,:,i);
       MA2=MA2+P(:,:,i)+m(:,:,i)*m(:,:,i)';
   end
   A(r,:)=MA1/(MA2+R(r,r)*diag(1./GaA(r,:)));
   end

   %% update B

   for r=1:n

       MB1=zeros(1,p);MB2=zeros(p,p);

   for i=1:T
       MB1=MB1+(m(r,:,i+1)-A(r,:)*m(:,:,i))*U(:,i)';
       MB2=MB2+U(:,i)*U(:,i)';
   end
       B(r,:)=MB1/(MB2+R(r,r)*diag(1./GaB(r,:)));
   end

   %% update C

   for r=1:mm

   MC1=zeros(1,n);MC2=zeros(n);

   for i=1:T
       MC1=MC1+(Y(r,i)-D(r,:)*U(:,i))*m(:,:,i+1)';
       MC2=MC2+P(:,:,i+1)+m(:,:,i+1)*m(:,:,i+1)';
   end
   C(r,:)=MC1/(MC2+Q(r,r)*diag(1./GaC(r,:)));
   end

   %% update D
   for r=1:mm

   MD1=zeros(1,p);MD2=zeros(p);

   for i=1:T
       MD1=MD1+(Y(r,i)-C(r,:)*m(:,:,i+1))*U(:,i)';
       MD2=MD2+U(:,i)*U(:,i)';
   end
       D(r,:)=MD1/(MD2+Q(r,r)*diag(1./GaD(r,:)));
   end

   %% update R

   for j=1:n
     MR1=0;MR2=0;
        for i=1:T
            Rup1=m(:,:,i+1)-A*m(:,:,i)-B*U(:,i);
            MR1=MR1+Rup1(j)^2;
            Rup2=P(:,:,i+1)-A*PC(:,:,i)-PC(:,:,i)*A'+A*P(:,:,i)*A';
            MR2=MR2+Rup2(j,j);
        end
      R(j,j)=MR1/T+MR2/T;
   end 

%% update Q
   for j=1:mm
     MQ1=0;MQ2=0;
        for i=1:T
            Qup1=Y(:,i)-C*m(:,:,i+1)-D*U(:,i);
            MQ1=MQ1+Qup1(j)^2;
            Qup2=C*P(:,:,i+1)*C';
            MQ2=MQ2+Qup2(j,j);
        end
     Q(j,j)=MQ1/T+MQ2/T;
   end 

   %% update GaA

   GaA=(A.^2+2*b0)/(2*a0+3);

   %% update GaB

   GaB=(B.^2+2*b0)/(2*a0+3);

   %% update GaC

   GaC=(C.^2+2*b0)/(2*a0+3);

   %% update GaD

   GaD=(D.^2+2*b0)/(2*a0+3);

   %% compute error

    error(1,ncount+1)=sum(sum((A2-A).^2));
    error(2,ncount+1)=sum(sum((B2-B).^2));
    error(3,ncount+1)=sum(sum((C2-C).^2));
    error(4,ncount+1)=sum(sum((D2-D).^2));
    error(5,ncount+1)=sum(sum((R2-R).^2));
    error(6,ncount+1)=sum(sum((Q2-Q).^2));

end




