forked from hashbrowncipher/doppler
-
Notifications
You must be signed in to change notification settings - Fork 0
/
multilateration.py
76 lines (58 loc) · 2.68 KB
/
multilateration.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
from itertools import combinations
from operator import itemgetter
import sys
from numpy import array
from numpy import random
from numpy import concatenate
from numpy import sum
from numpy import mean
from numpy.linalg.linalg import pinv
from numpy.linalg.linalg import LinAlgError
from scipy.spatial.distance import euclidean
from scipy.spatial.distance import cdist
from scipy.spatial.distance import pdist
from scipy.spatial.distance import squareform
from pipe_util import split_fileinput
from pipe_util import join_output
from pipe_util import MULTILATERATE_INPUT_FORMAT
from pipe_util import DISPLAY_INPUT_FORMAT
from world_params import CHANNEL_COUNT
from world_params import SAMPLE_RATE_HERTZ
from world_params import MIC_COORDS_METERS
FAKE_DART_METERS = array([1.5, 2, 0.5])
MIC_DART_DISTANCES_METERS = array([euclidean(mic, FAKE_DART_METERS) for mic in MIC_COORDS_METERS])
def transpose_1D(M):
"""numpy is silly and won't let you transpose a N x 1 ndarray to a 1 x N
ndarray."""
return M.reshape(len(M), 1)
def multilaterate(mic_positions, time__mic_dart_distances_stream):
"""Take a stream of mic - dart distances and yield coordinates.
mic_positions must be a a N x 3 array of coordinates of each mic.
time__mic_dart_distances_stream must be a generator of time + CHANNEL *
distances from mic to dart.
Yields time, 3D coords of dart.
"""
mic_positions = array(mic_positions)
origin = mic_positions[0]
mic_positions = mic_positions - origin
for time__mic_dart_distances in time__mic_dart_distances_stream:
time_seconds = time__mic_dart_distances[0]
mic_dart_distances = array(time__mic_dart_distances[1:])
# The algorithm fails on any 0 - m distance degeneracies. Add some
# wiggle if so.
degeneracies = (mic_dart_distances[1:] == mic_dart_distances[0])
mic_dart_distances[1:] += 1.0e-12 * degeneracies
vt = mic_dart_distances - mic_dart_distances[0]
free_vt = vt[2:]
A = 2.0 * mic_positions[2:, 0] / free_vt - 2.0 * mic_positions[1, 0] / vt[1]
B = 2.0 * mic_positions[2:, 1] / free_vt - 2.0 * mic_positions[1, 1] / vt[1]
C = 2.0 * mic_positions[2:, 2] / free_vt - 2.0 * mic_positions[1, 2] / vt[1]
D = free_vt - vt[1] - sum(mic_positions[2:, :] ** 2, axis=1) / free_vt + sum(mic_positions[1] ** 2) / vt[1]
M = concatenate([transpose_1D(A), transpose_1D(B), transpose_1D(C)], axis=1)
try:
yield time_seconds, pinv(M).dot(-transpose_1D(D)).reshape(3) + origin
except LinAlgError:
sys.stderr.write('Could not multilaterate at t = %f\n' % time_seconds)
if __name__ == '__main__':
for time_seconds, coordinates in multilaterate(MIC_COORDS_METERS, split_fileinput(MULTILATERATE_INPUT_FORMAT)):
join_output(DISPLAY_INPUT_FORMAT, [time_seconds] + list(coordinates))