function [f_vec,g_vec,time_vec,x,acc_vec] = AGM_BiO(fun_f,grad_f,grad_g,fun_g,TSA,param,x0)

% param definition
epsilon_f= param.epsilonf;
epsilon_g= param.epsilong;
L_f = param.L_f;
lambda = param.lam;
maxiter = param.maxiter;

n = length(x0);
x = x0;
z = x0;
A = 0;
% fun_g_x0 = param.fun_g_x0;
gstar = param.gstar;
% algorithm
iter = 0;
f_vec = [];
g_vec = [];
time_vec = [];
acc_vec = [];
% X_vec = [];
% Z_vec = [];
tic;
while iter <= maxiter
    % Descent step
%     fprintf('Iteration: %d\n',iter)
    a = (iter+1)/(4*L_f)/100;
    y = A/(A+a)*x + a/(A+a)*z;
%     disp(norm(z,2));
    z = z-a*grad_f(y);
%     disp(norm(z,2));
%     z = ProjectOntoL2Ball0(z, lambda);
    z = ProjectOntoIntersection(z,lambda,grad_g(y),grad_g(y)'*y+gstar-fun_g(y));
%     disp(norm(z,2));
    % Projection step
%     H = [eye(n) -eye(n);-eye(n) eye(n)];
%     h = [-z z];
% %     b=[grad_g(y)'*y+fun_g_x0-fun_g(y); lambda];
%     b=[grad_g(y)'*y+gstar-fun_g(y); lambda];
%     B = [grad_g(y)' -grad_g(y)'; ones(1,2*n)];
%     lb=[sparse(2*n,1)];
%     v = quadprog(H,h,B,b,[],[],lb,[]);
%     z = v(1:n)-v(n+1:end);
%     disp(size(x));
%     disp(size(z));
    x = A/(A+a)*x + a/(A+a)*z;
    A = A+a;
    iter = iter+1;
%     gamma = 2/(iter+2+10);
%     % Find direction
%     b=[grad_g(x)'*x+fun_g_x0-fun_g(x); lambda];
%     A = [grad_g(x)' -grad_g(x)'; ones(1,2*n)];
%     lb=[sparse(2*n,1)];
%     f=[grad_f(x)' -grad_f(x)'];
%     options = optimoptions('linprog','Algorithm','dual-simplex','Display','off');
%     % options = optimoptions('linprog','Algorithm','dual-simplex','Display','iter');
%     vec = linprog(f,A,b,[],[],lb,[],options);
%     d=vec(1:n)-vec(n+1:end);
%     if grad_f(x)'*(x-d)<=epsilon_f && (fun_g(x)-fun_g_x0)<=epsilon_g/2
%         break;
%     end
%     x = (1-gamma)*x + gamma*d;
    cpu_t = toc;
    f_vec = [f_vec;fun_f(x)];
    g_vec = [g_vec;fun_g(x)];
    time_vec = [time_vec;cpu_t];
    % test set accuracy
    acc_vec = [acc_vec;TSA(x)];
%     X_vec = [X_vec;x'];
%     Z_vec = [Z_vec;z'];
end

end