function amtmex(varargin)
%AMTMEX Compile Mex/Oct interfaces
% Usage: amtmex;
% amtmex(...);
%
% AMTMEX compiles the C backend in order to speed up the execution of
% the toolbox. The C backend is linked to Matlab and Octave through mex
% and Octave C++ interfaces.
%
% The action of AMTMEX is determined by one of the following flags:
%
% 'compile' Compile stuff. This is the default.
%
% 'clean' Removes the compiled functions.
%
% Url: http://amtoolbox.sourceforge.net/amt-0.9.7/doc/amtmex.php
% Copyright (C) 2009-2015 Piotr Majdak and Peter L. Søndergaard.
% This file is part of AMToolbox version 0.9.7
%
% 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 3 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, see <http://www.gnu.org/licenses/>.
% AUTHOR : Peter Søndergaard.
% TESTING: NA
% REFERENCE: NA
bp=mfilename('fullpath');
bp=bp(1:end-6);
defnopos.flags.command={'compile','clean'};
[flags,kv]=ltfatarghelper({},defnopos,varargin);
% Remember the current directory.
curdir=pwd;
if isoctave
extname='oct';
else
extname='mex';
end;
% -------------- Handle cleaning --------------------------------
if flags.do_clean
if ~isoctave
% Delete files permanently (bypass trashcan on Windows/Mac
% but remember the old state
oldstate = recycle('off');
end;
s=sprintf('========= Cleaning %s interfaces ==========', extname);
amtdisp(s);
if isoctave
deletefiles([bp,'oct'],'*.oct');
deletefiles([bp,'oct'],'*.o');
deletefiles([bp,'oct'],'*.mex');
else
deletefiles([bp,'mex'],['*.',mexext]);
end;
if ~isoctave
recycle(oldstate);
end;
end;
% -------------- Handle compiling --------------------------------
if flags.do_compile
s=sprintf('========= Compiling %s interfaces ==========', extname);
amtdisp(s);
if isoctave
ext='oct';
L=dir([bp,filesep,'oct',filesep,'*.cc']);
else
ext=mexext;
% Get the list of files.
L=dir([bp,filesep,'mex',filesep,'comp_*.c']);
L=[L; dir([bp,filesep,'mex',filesep,'comp_*.cpp'])];
end;
filenames = arrayfun(@(lEl) lEl.name,L,'UniformOutput',0);
if compile_amt(bp,ext,filenames)>1;
s=sprintf('Error: The %s interfaces was not built.', extname);
amtdisp(s);
else
amtdisp('Done.');
end;
if isoctave
% Compile MEXs instead of missing OCTs
Lmex=dir([bp,filesep,'mex',filesep,'comp_*.c']);
mexnamesstrip = arrayfun(@(lEl) lEl.name(1:end-2),Lmex,'UniformOutput',0);
octnamesstrip = cellfun(@(lEl) lEl(1:end-3),filenames,'UniformOutput',0);
mexdiffstrip = setdiff(mexnamesstrip,octnamesstrip);
mexdiff = cellfun(@(lEl) [lEl,'.c'],mexdiffstrip,'UniformOutput',0);
if ~isempty(mexdiff)
amtdisp('========= Compiling MEX interfaces ==========')
if compile_amt(bp,'mex',mexdiff)>1;
s=sprintf('Error: The %s interfaces was not built.', extname);
amtdisp(s);
else
if movefile([bp,filesep,'mex',filesep,'*.mex'],...
[bp,filesep,'oct'],'f');
amtdisp('Done.');
else
error(['Error: Compilation sucessful, but MEX files were not '...
'moved from mex to oct directory. Check your write permissions.\n']);
end
end;
end
end;
% Jump back to the original directory.
cd(curdir);
end
function deletefiles(base,files)
L=dir([base,filesep,files]);
for ii=1:numel(L)
s=[base,filesep,L(ii).name];
delete(s);
end;
function status=compile_amt(bp,ext,filenames)
% If we exit early, it is because of an error, so set status=1
status=1;
if strcmpi(ext(1:3),'oct')
cd([bp,'oct']);
else
cd([bp,'mex']);
end
for ii=1:numel(filenames)
filename = filenames{ii};
dotPos = strfind(filename,'.');
objname = [filename(1:dotPos(end)),ext];
objdirinfo = dir(objname);
% Make-like behaviour: build only the files where the src file is
% newer than the object file, or the object file is missing.
L = dir(filename);
if isempty(objdirinfo) || (objdirinfo.datenum<L(1).datenum)
s=sprintf('Compiling %s',filename);
amtdisp(s);
if isoctave
if ~strcmpi(ext(1:3),'oct')
mkoctfile('-mex','-I.','-I../src',filename);
else
mkoctfile('-I.','-I../src',filename);
end
else
mex('-I.','-I../src',filename);
end;
end;
status=0;
end;