-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathphg_overlap_conv.m
52 lines (42 loc) · 1.48 KB
/
phg_overlap_conv.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
function y=phg_overlap_conv(x,h,k)
% calcula a convolução através do método overlap-add
% x e h são 2 vetores de qualquer forma, e k é o número de elementos em
% cada bloco, retorna vetor coluna
% Overlap-add é uma forma eficiente para calcular a convolução discreta de
% um sinal x longo com uma resposta ao impulso finita;
% dividindo-se o sinal em blocos de tamanho apropriado, podemos calcular
% sua convolução discreta com apenas um pequeno delay.
% sobre: https://en.wikipedia.org/wiki/Overlap%E2%80%93add_method
% se alguns dos argumentos possuir apenas 1 elementos, a convolução é o
% produto escalar entre eles.
if min([numel(x) numel(h) k]) == 1
y = phg_conv(x,h);
return
end
% x precisa ser inteiramente dividido em blocos de k elementos, por isso
% adicionamos k zeros ao final de x
x(numel(x)+k)=0;
% L é o tamanho de x, com zeros adicionados, M é o tamanho de y
M = numel(h);
L = numel(x);
% protege contra vetores linha, pois y e x precisam ser colunas
if (isrow(x)==1)
x = transpose(x);
end
if (isrow(h) == 1)
h = transpose(h);
end
% pré-aloca y_aux, que é a matrix com as convoluções deslocadas
y = zeros(L+M-1);
% calcula uma matriz que é a convolução entre h, e os pedaçoes do vetor x,
% de tamanho k
for n = 1:floor(L/k)
y = y + [zeros((n-1)*k,1); phg_conv( x(1+(n-1)*k:n*k), h ); zeros(L-n*k,1)];
end
% pré aloca y, que é a convolução
% y = zeros( L+M-1 );
% % soma todas as colunas da matriz y_aux
% for n = 1:floor(L/k)
% y = y+y_aux(:,n);
% end
y=y(1:L+M-k-1);