% This is the testing demo of FFDNet for denoising noisy color images corrupted by
% AWGN.
%
% To run the code, you should install Matconvnet first. Alternatively, you can use the
% function `vl_ffdnet_matlab` to perform denoising without Matconvnet.
%
% "FFDNet: Toward a Fast and Flexible Solution for CNN based Image
% Denoising" 2018/03/23
% If you have any question, please feel free to contact with me.
% Kai Zhang (e-mail: cskaizhang@gmail.com)

% clear; clc;

format compact;
global sigmas; % input noise level or input noise level map
addpath(fullfile('utilities'));

folderModel = 'models';
folderTest  = 'testsets';
folderResult= 'results';
imageSets   = {'CBSD68','Kodak24','McMaster'}; % testing datasets
setTestCur  = imageSets{1};      % current testing dataset

showResult  = 1;
useGPU      = 0;
pauseTime   = 0;

imageNoiseSigma = 25;  % image noise level
inputNoiseSigma = 25;  % input noise level

folderResultCur       =  fullfile(folderResult, [setTestCur,'_',num2str(imageNoiseSigma(1)),'_',num2str(inputNoiseSigma(1))]);
if ~isdir(folderResultCur)
    mkdir(folderResultCur)
end

load(fullfile('models','FFDNet_color.mat'));%fullfileº¯Êý×÷ÓÃÊÇ×÷ÓÃÊÇÀûÓÃÎÄ¼þ¸÷²¿·ÖÐÅÏ¢´´½¨²¢ºÏ²¢³ÉÍêÕûÎÄ¼þÃû£º
%load net £ºÕâÊÇÔ¤ÏÈÑµÁ·ºÃµÄÒ»¸öÍøÂç
%½«Æä±äÎªsimplennµÄÍøÂç
%matconvnetÓÐÁ½ÖÖÍøÂç£º»¹ÓÐÒ»ÖÖÎªDAG Ä£ÐÍ£¬
% Á½¸öÍøÂçµÄ²»Í¬Ö®´¦ÔÚÓÚ½«ÍøÂçÒÔ²»Í¬µÄÐÎÊ½ÏÔÊ¾³öÀ´£¬ºóÕßDAG »á¸üÖ±¹Û
net = vl_simplenn_tidy(net);

% for i = 1:size(net.layers,2)
%     net.layers{i}.precious = 1;
% end

if useGPU
    net = vl_simplenn_move(net, 'gpu') ;%vl_simplenn_move ½«CNNÔÚCPUºÍGPUÖ®¼äÒÆ¶¯
end

% read images
ext         =  {'*.jpg','*.png','*.bmp','*.tif'};
filePaths   =  [];
for i = 1 : length(ext)
    filePaths = cat(1,filePaths, dir(fullfile(folderTest,setTestCur,ext{i})));
    %cat(1, A, B)Ïàµ±ÓÚ[A; B].
end
%°Ñ./testsets/CBSD68/ÎÄ¼þ¼ÐÏÂ£¬ËùÓÐÂú×ãºó×ºÒªÇóµÄ¶¼·ÅÔÚfilePathsÖÐ¡£

% PSNR and SSIM
PSNRs = zeros(1,length(filePaths));
SSIMs = zeros(1,length(filePaths));

for i = 1:length(filePaths)%¶ÔÃ¿Ò»ÕÅÍ¼Æ¬µÄ±éÀú length(filePaths)=68
    
    % read images
    %          imread ='testsets\CBSD68\101085.png'
    label   = imread(fullfile(folderTest,setTestCur,filePaths(i).name));
    [w,h,c] = size(label); % 481*321*3

    
    if c == 3 %²ÊÉ«Í¼
        [~,nameCur,extCur] = fileparts(filePaths(i).name); 
        %fileparts ¸Ãº¯ÊýÓÃÓÚ½«Ò»¸öÎÄ¼þµÄÍêÕûÂ·¾¶ÖÐ¸÷²¿·ÖÌáÈ¡³öÀ´¡£ÕâÒ»²½ÓÐÉ¶ÓÃÄØ
        label = im2double(label);
        
        % add noise
        randn('seed',0);
        noise = bsxfun(@times,randn(size(label)),permute(imageNoiseSigma/255,[3 4 1 2]));
        %permute£ºÖØÐÂÅÅÁÐ¾ØÕóµÄÎ¬¶È£¬°Ñ3 4Î¬ÌáÇ°
        input = single(label + noise);
        %Èç¹ûÄúÓÐÒ»¸öÊý×éÓÉ²»Í¬µÄÊý¾ÝÀàÐÍ£¨Èç double »ò int8£©×é³É£¬Ôò¿ÉÒÔÊ¹ÓÃ single º¯Êý½«¸ÃÊý×é×ª»»Îªµ¥¾«¶ÈÊý×é¡£
        
        if mod(w,2)==1 %¶ÔÐÐÈ¡Ä££¬Èç¹ûÐÐÊý²»ÄÜ±»2Õû³ý£¬¾Í°Ñ×îºóÒ»ÐÐµÄÇÐÆ¬ÖØ¸´¼ÓÔÚinputµÄÄ©Î²
            input = cat(1,input, input(end,:,:)) ;
        end
        if mod(h,2)==1%¶ÔÁÐÈ¡Ä£
            input = cat(2,input, input(:,end,:)) ;
        end
      %×îÖÕÐ§¹û  size(input)= 482   322     3
        % tic;
        if useGPU
            input = gpuArray(input); %ÀûÓÃgpuArray()º¯Êý½«Êý¾Ý´ÓCPU´«ÈëGPUÖÐ
        end
        %Êý¾ÝÔÚ½øÐÐÔËËãÊ±£¬Ö»ÒªÓÐÒ»¸ö±äÁ¿ÔÚGPUÉÏ£¬ÆäËû±äÁ¿Ò²»á×Ô¶¯½øÈë GPUÒ»ÆðÔËËã£¬²úÉúµÄ½á¹ûÒ²ÔÚGPUÉÏ¡£
        %Èç¹ûÐèÒªµÄ»°£¬ÓÃgather()º¯Êý½«Êý¾Ý´ÓGPU´«»ØCPU
        
        % set noise level map
        sigmas = inputNoiseSigma/255; % see "vl_simplenn.m".
        
        % perform denoising
        %res    = vl_simplenn(net,input,[],[],'conserveMemory',true,'mode','test'); % matconvnet default
        % res    = vl_ffdnet_concise(net, input);    % concise version of vl_simplenn for testing FFDNet
        res    = vl_ffdnet_matlab(net, input); % use this if you did  not install matconvnet; very slow
        
        % output = input -res(end).x; % for 'model_color.mat'
        output = res(end).x;
        
        
        if mod(w,2)==1 %ÔÙ°Ñ¸Õ²Å¼ÓµÄ×îºóÒ»ÐÐºÍÒ»ÁÐÉáÈ¥
            output = output(1:end-1,:,:);
            input  = input(1:end-1,:,:);
        end
        if mod(h,2)==1
            output = output(:,1:end-1,:);
            input  = input(:,1:end-1,:);
        end
        
        if useGPU
            output = gather(output); %Èç¹ûÐèÒªµÄ»°£¬ÓÃgather()º¯Êý½«Êý¾Ý´ÓGPU´«»ØCPU
            input  = gather(input);
        end
        %toc;
        
        % calculate PSNR, SSIM and save results
        [PSNRCur, SSIMCur] = Cal_PSNRSSIM(im2uint8(label),im2uint8(output),0,0);
        if showResult
            imshow(cat(2,im2uint8(input),im2uint8(label),im2uint8(output)));
            title([filePaths(i).name,'    ',num2str(PSNRCur,'%2.2f'),'dB','    ',num2str(SSIMCur,'%2.4f')])
            %imwrite(im2uint8(output), fullfile(folderResultCur, [nameCur, '_' num2str(imageNoiseSigma(1),'%02d'),'_' num2str(inputNoiseSigma(1),'%02d'),'_PSNR_',num2str(PSNRCur*100,'%4.0f'), extCur] ));
            drawnow;
            pause()
        end
        disp([filePaths(i).name,'    ',num2str(PSNRCur,'%2.2f'),'dB','    ',num2str(SSIMCur,'%2.4f')])
        PSNRs(i) = PSNRCur;
        SSIMs(i) = SSIMCur;
        
    end
end

disp([mean(PSNRs),mean(SSIMs)]);




