-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsource.py
130 lines (111 loc) · 3.53 KB
/
source.py
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
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
"""Compute impulse responses of source models."""
import numpy as np
from scipy.signal import remez, fftconvolve
from sys import path
path.append('../')
from utils import *
def greens_plane(npw, x, c=343):
"""Greens function of a plane wave.
Parameters
----------
npw : (3,) array_like
Plane wave (propagating) direction
x : (N, 3) array_like
Receiver positions in the Cartesian coordinate [m]
c : float
Speed of sound [m/s]
Returns
-------
delay : (N,) array_like
Delay with respect to the origin [s]
weight : (N,) array_like
Attenuation
"""
npw = np.array(npw) / np.linalg.norm(npw)
x = np.array(x)
if x.shape[1] != 3 & x.shape[0] == 3:
x = x.T
delay = x.dot(npw)/c
# delay -= np.min(delay)
return delay, 1
def greens_point(xs, x, c=343):
"""Greens function of a point source.
Parameters
----------
xs : (3,) array_like
Source position in the Cartesian coordiate [m]
x : (N, 3) array_like
Receiver positions in the Cartesian coordinate [m]
c : float
Speed of sound [m/s]
Returns
-------
delay : (N,) array_like
Propagation delay [s]
weights : (N,) array_like
Attenuation
"""
xs = np.array(xs)
x = np.array(x)
if x.ndim > 1:
if x.shape[1] != 3 & x.shape[0] == 3:
x = x.T
distance = np.linalg.norm(x - xs[np.newaxis, :], axis=-1)
return distance/c, 1/4/np.pi/distance
def impulse_response(xs, x, sourcetype, fs, oversample=2, c=343):
"""Impulse responses for a given source type.
Parameters
----------
xs : (3,) array_like
Source position in the Cartesian coordinate [m]
x : (N, 3) array_like
Receiver positions in the Cartesian coordinate [m]
sourcetype : string
Source type e.g. 'point'
fs : int
Sampling frequency [Hz]
oversample : int, optional
Oversampling factor
c : float, optional
Speed of sound
Returns
-------
waveform : (N, C) array_like
Waveforms (nonzero coefficients) of the impulse resposnes
shift : (N,) int, array_like
Shift (number of preceeding zeros) [sample]
offset : (N,) int, array_like
Position of the main peak [sample]
"""
if sourcetype == 'point':
delay, weight = greens_point(xs, x, c)
elif sourcetype == 'plane':
delay, weight = greens_plane(xs, x, c)
weight = weight * np.ones_like(delay)
else:
print('sourcetype error')
waveform_up, shift_up, offset_up = fractional_delay(delay, Lf=23, fs=oversample*fs, type='fast_lagr')
f = fir_minmax(fs, oversample)
filtorder = int((len(f)+1)/2)
waveform_up = fftconvolve(waveform_up, f[np.newaxis, :])
shift_up -= filtorder
offset_up -= filtorder
shift = shift_up // oversample
res = shift_up % oversample
offset = offset_up // oversample
waveform_up = np.column_stack((waveform_up, np.zeros((len(waveform_up), oversample-1))))
for n in range(len(shift)):
waveform_up[n, :] = np.roll(waveform_up[n, :], res[n])
waveform = waveform_up[:, ::oversample]
return waveform * weight[:, np.newaxis], shift, offset
def fir_minmax(fs, Q):
"""Low-pass filter for sampling rate conversion."""
fpass = 0.8 * fs / 2
fstop = 1.0 * fs / 2
att = -130
filtorder = Q * 40
if Q != 1:
f = remez(2*filtorder+1, [0, fpass, fstop, Q*fs/2], [1, 10**((att)/20)], weight=[1, 10e5], fs=Q*fs)
else:
f = np.array(1)
return f