function [bm,env,GFB] = may2011_gammatone(in,fs,fLow,fUp,nFilter,bEar,bAlign,bInfo)
%MAY2011_GAMMATONE Auditory filterbank
%
% Usage:
% [BM,ENV,GFB] = may2011_gammatone(IN)
% [BM,ENV,GFB] = may2011_gammatone(IN,GFB)
% [BM,ENV,GFB] = may2011_gammatone(IN,FS,FLOW,FUP,NFILTER,bEAR,bALIGN,bINFO)
%
% Input parameters:
% IN : audio object. The gammatone parameter will be initialized
% using the audio sampling frequency.
% IN : mono input signal [nSamples x nChannels]
% GFB : gammatone parameter structure (see gammatoneInit)
% IN : multi-channel input signal [nSamples x nChannels]
% FS : sampling frequency in Hz
% FLOW : center frequency of lowest auditory filter
% (default, FLOW = 80)
% FUP : center frequency of highest auditory filter
% (default, FUP = 5e3)
% NFILTER : number of auditory filters which will be spaced linear in
% the ERB domain. If NFILTER is not a scalar but a vector, the
% first value is assumed to represent the number of auditory
% channels and the following values represents the indices of
% the filters which should be processed. This can be useful if
% a large number of channels is required as MATLAB might run
% out of memory for long signals if all filters should be
% computed in one step.
% (default, NFILTER = round(freq2erb(FS/2))
% bEAR : Adjust gain coefficients of the auditory channels to
% incorporate middle ear effects. Note that this feature can
% only be used if the center frequencies of the auditory
% channels are above 20 Hz and below 12500 Hz.
% (default, bEAR = true)
% bALIGN : phase-aligned gammatone output (non-causal output)
% (default, bALIGN = false)
% bINFO : info flag printing gammatone parameters on the screen
% (default, bINFO = false)
%
% Output parameters:
% BM : basilar membrane displacement [nSamples x nFilter]
% ENV : instantaneous envelope [nSamples x nFilter]
% GFB : gammatone parameter structure
%
% MAY2011_GAMMATONE applies gammatone filtering using the gammatone
% structure derived from gammatone_init. If only the basilar membrane
% output "bm" is required, computing BM = gammatone(...);
% will be significantly faster then computing the envelope as well:
% [BM,ENV] = gammatone(...);
%
% The MEX implementaion is based on the source code of the ohio-state
% university: www.cse.ohio-state.edu/pnl/shareware/roman-jasa03/
%
% Example:
%
% nSamples = 500;
% % Initialize gammatone parameter structure
% GFB = may2011_gammatone_init(20e3);
% % Filter impulse with gammatone filtering
% bm = may2011_gammatone([1; zeros(nSamples-1,1)],GFB);
% % Plot result
% waveplot(1:nSamples,GFB.cf,bm);
%
% See also: may2011_gammatoneinit
%
% Url: http://amtoolbox.org/amt-1.6.0/doc/modelstages/may2011_gammatone.php
% #StatusDoc: Good
% #StatusCode: Good
% #Verification: Unknown
% #Requirements: MATLAB M-Signal
% #Author: Tobias May (2014)
% This file is licensed unter the GNU General Public License (GPL) either
% version 3 of the license, or any later version as published by the Free Software
% Foundation. Details of the GPLv3 can be found in the AMT directory "licences" and
% at <https://www.gnu.org/licenses/gpl-3.0.html>.
% You can redistribute this file and/or modify it under the terms of the GPLv3.
% This file is distributed without any warranty; without even the implied warranty
% of merchantability or fitness for a particular purpose.
% Check for proper input arguments
if nargin < 2 || nargin > 8
help(mfilename);
error('Wrong number of input arguments!');
end
% If two parameters are supplied, check if the second argument is a
% gammatone filterbank structure...
if nargin == 2 && isGFB(fs);
% Set initialization flag to true
bInitOK = true;
else
% Initialization not completed yet...
bInitOK = false;
end
% Check initialization flag
if bInitOK
% Copy gammatone filterbank structure
GFB = fs;
else
% Set default values
if nargin < 3 || isempty(fLow); fLow = 80; end
if nargin < 4 || isempty(fUp); fUp = min(5e3,fs/2); end
if nargin < 5 || isempty(nFilter); nFilter = round(freq2erb(fs/2)); end
if nargin < 6 || isempty(bEar); bEar = false; end
if nargin < 7 || isempty(bAlign); bAlign = false; end
if nargin < 8 || isempty(bInfo); bInfo = false; end
% Initialize gammatone filterbank structure
GFB = may2011_gammatoneinit(fs,fLow,fUp,nFilter,bEar,bAlign,bInfo);
end
% Check for consistent gammatone channel selection
if length(GFB.filter2Process) > 1 && ...
GFB.filter2Process(1) < max(GFB.filter2Process(2:end))
error(['Inconsistent channel selection. nFilter(1) represents ' ,...
'the total number auditory filters which are linearly ' ,...
'spaced on the ERB axis, whereas the remaining indices ' ,...
'nFilter(2:end) correspond to the auditory channels ' ,...
'which should be processed.'])
end
% Determine data size
[nSamples,nChannels] = size(in);
% TODO % Enable chunk-based processing by keeping track of filter states
% Check number of output arguments
if nargout > 1
% =====================================================================
% Envelope required
% =====================================================================
%
% Allocate memory
[bm,env]=deal(zeros(nSamples,length(GFB.filter2Process)-1,nChannels));
% Loop over number of audio channels
for ii = 1 : nChannels
% replace the original MEX filename by the AMT MEX filename
if strcmp(GFB.fcnHandle,'gammatoneMEX'), GFB.fcnHandle='comp_may2011_gammatone'; end
% Call gammatone filterbank routine (MEX-file implementation)
[bm(:,:,ii),env(:,:,ii)] = feval(GFB.fcnHandle,in(:,ii),GFB.fs,...
GFB.lowerFreq,GFB.upperFreq,...
GFB.filter2Process,...
GFB.bOuterMiddleEar,...
GFB.bPhaseAlign,GFB.bInfo);
end
else
% =====================================================================
% Compute basilar membrane response only
% =====================================================================
%
% Allocate memory
bm = zeros(nSamples,length(GFB.filter2Process)-1,nChannels);
% Loop over number of audio channels
for ii = 1 : nChannels
% replace the original MEX filename by the AMT MEX filename
if strcmp(GFB.fcnHandle,'gammatoneMEX'), GFB.fcnHandle='comp_may2011_gammatone'; end
% Call gammatone filterbank routine (MEX-file implementation)
bm(:,:,ii) = feval(GFB.fcnHandle,in(:,ii),GFB.fs,...
GFB.lowerFreq,GFB.upperFreq,...
GFB.filter2Process,GFB.bOuterMiddleEar,...
GFB.bPhaseAlign,GFB.bInfo);
end
end
% TODO % Adjust outer/middle ear gain here to allow various weightings
function out = isGFB(in)
%isGFB Check if input is a gammatone filterbank structure.
% This is a small helper function in order to check if gammatone
% filterbank is initialized properly.
%
%USAGE
% OUT = isGFB(IN)
%
%INPUT ARGUMENTS
% IN : input
%
%OUTPUT ARGUMENTS
% OUT : true/false depending on whether IN is a gammatone structure
%
%EXAMPLE
% % Create gammatone structure
% p = gammatoneInit(44.1e3);
% % Check
% isGFB(p)
%ans =
% 1
%
% See also gammatone.
% Developed with Matlab 7.4.0.287 (R2007a). Please send bug reports to:
%
% Author : Tobias May, 2008
% TUe Eindhoven and Philips Research
% t.may@tue.nl tobias.may@philips.com
%
% History :
% v.0.1 2008/05/11
% ***********************************************************************
% Check for proper input arguments
if nargin ~= 1
help(mfilename);
error('Wrong number of input arguments!');
end
% Initialize output
out = false;
% Check if IN is a structure
if isstruct(in)
% Required structure fields
reqFields = {'fcnHandle' 'fs' 'lowerFreq' 'upperFreq' ...
'filter2Process' 'bOuterMiddleEar' 'bPhaseAlign' 'bInfo'};
% Check if all required fields are present
if all(isfield(in,reqFields))
% Set flag to true
out = true;
end
end