function [X,errors,times] = TNN(M,omega,X_star,opts)

tol = 1e-13; 
max_iter =opts.max_iter;
rho = 1.1;
mu = 1e-4;
max_mu = 1e10;
DEBUG = 0;

if ~exist('opts', 'var')
    opts = [];
end    
if isfield(opts, 'tol');         tol = opts.tol;              end
if isfield(opts, 'max_iter');    max_iter = opts.max_iter;    end
if isfield(opts, 'rho');         rho = opts.rho;              end
if isfield(opts, 'mu');          mu = opts.mu;                end
if isfield(opts, 'max_mu');      max_mu = opts.max_mu;        end
if isfield(opts, 'DEBUG');       DEBUG = opts.DEBUG;          end

dim = size(M);
X = zeros(dim);
X(omega) = M(omega);
E = zeros(dim);
Y = E;

errors = [];
times  = [];
tcount = 0;

iter = 0;
for iter = 1 : max_iter
    t1 = tic;
    Xk = X;
    Ek = E;
    % update X
    [X,tnnX] = prox_tnn(-E+M+Y/mu,1/mu); 
    % update E
    E = M-X+Y/mu;
    E(omega) = 0;
 
    dY = M-X-E;    
    chgX = max(abs(Xk(:)-X(:)));
    chgE = max(abs(Ek(:)-E(:)));
    chg = max([chgX chgE max(abs(dY(:)))]);
    if DEBUG
        if iter == 1 || mod(iter, 10) == 0
            obj = tnnX;
            err = norm(dY(:));
            disp(['iter ' num2str(iter) ', mu=' num2str(mu) ...
                    ', obj=' num2str(obj) ', err=' num2str(err)]); 
        end
    end
    
    if chg < tol
        break;
    end 
    Y = Y + mu*dY;
    mu = min(rho*mu,max_mu);
    error = norm(X(:)-X_star(:))^2/norm(X_star(:))^2;
    errors = [errors,error];
    tcount = tcount + toc(t1);
    times = [times,tcount];
end


obj = tnnX;
err = norm(dY(:));

end