Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

Salve Habr. Probabiliter omnes qui propinquos vel amicos in plano invenit vel viderit usus est servitii gratuiti Flightradar24. Haec via commodissima est ad indagare situm reali temporis aircraft.

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

В in prima parte, Principium operans talis servitii online descriptus est. Nunc pergamus et figuramus quid mitti et recipi e aircraft notitia ad stationem recipiendam et ad ipsum Pythone decocendam nos perspiciamus.

historia

Patet, elit data non traducitur utentibus ut in eorum Suspendisse potenti. Systema vocatur ADS-B (dependens custodiae automaticae-passim), et adhibetur ut informationes automatice tradendae de aircraft ad centrum moderandum - eius identifier, coordinatae, directionis, velocitatis, altitudinis et aliarum notitiarum transmittantur. Ante adventum talium systematum mittentes punctum in ligula non videre potuerunt. Hoc satis iam non erat, cum multa essent plana.

Technice, ADS-B in aircraft transfusori consistit, qui intervenit notitiarum fasciculos in satis alta frequentia 1090 MHz mittit (alii modi sunt, sed non ita interest, quia coordinatae tantum hic transmittuntur). Quidem, praeter transfusorem, est etiam alicubi receptaculum in elit, sed nobis, ut users, receptaculum nostrum interest.

Obiter ad comparationem primum talis ratio, Airnav Radarbox, usoribus ordinariis destinata, in 2007 apparuit, et circa $ 900 constat, subscriptione ad officia retis alia $250 in anno constant.

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

Recognitiones illorum primorum dominorum Russorum in foro legi possunt radioscanner. Nunc receptores RTL-SDR crees facti sunt, simili fabrica $30 pro $conveniri potest, plura de hoc in in prima parte,. Ad ipsum protocollum transeamus - videamus quomodo operatur.

accepto annuit

Primum signum notari debet. Totum signum durationis tantum 120 microseconds habet, ut membra ejus commode disgreget, receptor SDR cum sampling frequentia saltem 5 MHz expetenda est.

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

Post recordationem, fasciculum WAV accipimus cum sampling rate of 5000000 exemplaria / sec, 30 seconds talium recordationis "pondere" circiter 500MB. Audire eam cum lusoriis instrumentis, vanum sane est - tabella sonum non continet, sed signum radiophonicum directe acceptum - hoc prorsus modo Software Radiophonia definita operatur.

Aperiemus et expediemus tabella utens Pythone. Qui experiri in seipso volunt, exemplum memoriae extrahere possunt Link.

Lets download tabella et vide quid intus est.

from scipy.io import wavfile
import matplotlib.pyplot as plt
import numpy as np

fs, data = wavfile.read("adsb_20190311_191728Z_1090000kHz_RF.wav")
data = data.astype(float)
I, Q = data[:, 0], data[:, 1]
A = np.sqrt(I*I + Q*Q)

plt.plot(A)
plt.show()

Consequuntur: videmus manifestas "pulsus" in curriculo sonitus.

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

Quisque "pulsus" signum est, cuius structura clare conspicitur si solutionem in grapho augeas.

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

Ut videre potes, pictura satis congruit cum iis quae in descriptione supra scripta sunt. Potes incipere dispensando data.

Decoding

Primum, fluminis frenum accipere debes. Signum ipsum Encoded utens Manchester descriptam;

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

A gradu differentiae sensim adpetentes facile est acquirere realem "0" et "1".

    bits_str = ""
    for p in range(8):
        pos = start_data + bit_len*p
        p1, p2 = A[pos: pos + bit_len/2], A[pos + bit_len/2: pos + bit_len]
        avg1, avg2 = np.average(p1), np.average(p2)
        if avg1 < avg2:
            bits_str += "0"
        elif avg1 > avg2:
            bits_str += "1"

Signi structura haec est:

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

Intueamur agros planius.

DF (Forma downlink, 5 bits) - genus relatum decernit. Plura sunt genera:

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol
(fons mensa)

Tantum interest in typus DF17, quia... Hoc est quod coordinatas aircraft continet.

ICAO (24 bits) - unicum codicem internationalis aircraft. Reprehendo potes planum per codicem Online (proh dolor, auctor adaequare desiit datorum, sed adhuc pertinet). Exempli gratia in codice 3c5ee2 sequenti notitia habetur:

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

Edit: in commento ad articulum Codicis ICAO descriptio accuratius traditur, commendo ut ea, quorum interest, legat.

DATA (56 vel 112 bis) — ipsa notitia quam decode volumus. Primum V frusta data sunt agri Typus Codecontinens subtypum notitiarum repositum (non confundendum cum DF). Horum generum admodum pauca sunt;

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol
(fons mensa)

Inspiciamus pauca exempla fasciculorum.

Aircraft idem

Exemplum in forma binaria:

00100 011 000101 010111 000111 110111 110001 111000

Data agris:

+------+------+------+------+------+------+------+------+------+------+
| TC,5 | EC,3 | C1,6 | C2,6 | C3,6 | C4,6 | C5,6 | C6,6 | C7,6 | C8,6 |
+------+------+------+------+------+------+------+------+------+------+

TC = 00100b = 4, singulae characteris C1-C8 codices continentes indices in linea correspondentes:
#ABCDEFGHIJKLMNOPQRSTUVWXYZ #####_###################################################

Per chordas decocendas facile est codicem aircraft adipisci: EWG7184

symbols = "#ABCDEFGHIJKLMNOPQRSTUVWXYZ#####_###############0123456789######"
code_str = ""
for p in range(8):
     c = int(bits_str[8 + 6*p:8 + 6*(p + 1)], 2)
     code_str += symbols[c]
print("Aircraft Identification:", code_str.replace('#', ''))

Airborne loco

Si nomen simplex est, coordinatae magis implicatae sunt. Traduntur in forma 2, paribus ac imparibus tabulae. Agrum codicem TC = 01011b = XI.

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol

Exemplum paris et imparis facis:

01011 000 000101110110 00 10111000111001000 10000110101111001
01011 000 000110010000 01 10010011110000110 10000011110001000

Calculus ipse coordinatarum occurrit secundum formulam magis technam;

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol
(fons)

GIS peritus non sum, unde sit nescio. Quis scit, scribe in comment.

Altitudo simplicior consideratur - secundum peculiarem partem, repraesentari potest ut vel multiplex 25 vel 100 pedum.

Airborne Velocitas

Sarcina cum TC=19. Interestinger hic res est quod celeritas potest esse vel accurate, ad terram (Ground Speed), vel aerium, quod mensuratur per sensorem aircraft (Airspeed). Multae praeterea agri transmittuntur;

Flightradar24 - quomodo operatur? Pars II, ADS-B protocol
(fons)

conclusio,

Ut videre potes, technologia ADS-B facta est symbiosis interesting, cum norma non solum doctoribus utilis est, sed etiam usoribus communibus utilis. Sed sane, munus praecipuum in hac technologia vilis receptorum digitalium SDR geritur, quae machinam ad litteram recipiendi significationibus frequentiis supra gigahertz "pro denariis."

In ipso quidem vexillo multo plus est. Eorum quorum interest videre potes PDF in pagina ICAO aut visitare iam dictum est website.

Verisimile est omnia praedicta multis usui esse, sed idea generalis saltem quomodo operatur, ut spero, manet.

Viam in Pythone decoder parata iam existit, studere potes hic. Et possessores receptorum SDR convenire possunt et praeparatum ads-B decoder emittere ex pagina, hoc fusius tractatum est in prima parte,.

Fons codicis parser in articulo infra incisa descriptus est. Hoc exemplum est experimentum quod non fingit esse productionem, sed quaedam operantur in eo, et adhiberi potest ad parse tabella supra scripta.
Source codice (Python)

from __future__ import print_function
from scipy.io import wavfile
from scipy import signal
import matplotlib.pyplot as plt
import numpy as np
import math
import sys
def parse_message(data, start, bit_len):
max_len = bit_len*128
A = data[start:start + max_len]
A = signal.resample(A, 10*max_len)
bits = np.zeros(10*max_len)
bit_len *= 10
start_data = bit_len*8
# Parse first 8 bits
bits_str = ""
for p in range(8):
pos = start_data + bit_len*p
p1, p2 = A[pos: pos + bit_len/2], A[pos + bit_len/2: pos + bit_len]
avg1, avg2 = np.average(p1), np.average(p2)
if avg1 < avg2:
bits_str += "0"
elif avg1 > avg2:
bits_str += "1"
df = int(bits_str[0:5], 2)
# Aircraft address (db - https://junzis.com/adb/?q=3b1c5c )
bits_str = ""
for p in range(8, 32):
pos = start_data + bit_len * p
p1, p2 = A[pos: pos + bit_len / 2], A[pos + bit_len / 2: pos + bit_len]
avg1, avg2 = np.average(p1), np.average(p2)
if avg1 < avg2:
bits_str += "0"
elif avg1 > avg2:
bits_str += "1"
# print "Aircraft address:", bits_str, hex(int(bits_str, 2))
address = hex(int(bits_str, 2))
# Filter specific aircraft (optional)
# if address != "0x3c5ee2":
#    return
if df == 16 or df == 17 or df == 18 or df == 19 or df == 20 or df == 21:
# print "Pos:", start, "DF:", msg_type
# Data (56bit)
bits_str = ""
for p in range(32, 88):
pos = start_data + bit_len*p
p1, p2 = A[pos: pos + bit_len/2], A[pos + bit_len/2: pos + bit_len]
avg1, avg2 = np.average(p1), np.average(p2)
if avg1 < avg2:
bits_str += "0"
# bits[pos + bit_len / 2] = 50
elif avg1 > avg2:
bits_str += "1"
# http://www.lll.lu/~edward/edward/adsb/DecodingADSBposition.html
# print "Data:"
# print bits_str[:8], bits_str[8:20],  bits_str[20:22], bits_str[22:22+17], bits_str[39:39+17]
# Type Code:
tc, ec = int(bits_str[:5], 2), int(bits_str[5:8], 2)
# print("DF:", df, "TC:", tc)
# 1 - 4  Aircraft identification
# 5 - 8  Surface position
# 9 - 18  Airborne position (w/ Baro Altitude)
# 19  Airborne velocities
if tc >= 1 and tc <= 4: # and (df == 17 or df == 18):
print("Aircraft address:", address)
print("Data:")
print(bits_str[:8], bits_str[8:14],  bits_str[14:20], bits_str[20:26], bits_str[26:32], bits_str[32:38], bits_str[38:44])
symbols = "#ABCDEFGHIJKLMNOPQRSTUVWXYZ#####_###############0123456789######"
code_str = ""
for p in range(8):
c = int(bits_str[8 + 6*p:8 + 6*(p + 1)], 2)
code_str += symbols[c]
print("Aircraft Identification:", code_str.replace('#', ''))
print()
if tc == 11:
print("Aircraft address:", address)
print("Data: (11)")
print(bits_str[:8], bits_str[8:20],  bits_str[20:22], bits_str[22:22+17], bits_str[39:39+17])
# Bit 22 contains the F flag which indicates which CPR format is used (odd or even)
# First frame has F flag = 0 so is even and the second frame has F flag = 1 so odd
# f = bits_str[21:22]
# print("F:", int(f, 2))
# Altitude
alt1b = bits_str[8:20]
if alt1b[-5] == '1':
bits = alt1b[:-5] + alt1b[-4:]
n = int(bits, 2)
alt_ft = n*25 - 1000
print("Alt (ft)", alt_ft)
# lat_dec = int(bits_str[22:22+17], 2)
# lon_dec = int(bits_str[39:39+17], 2)
# print("Lat/Lon:", lat_dec, lon_dec)
# http://airmetar.main.jp/radio/ADS-B%20Decoding%20Guide.pdf
print()
if tc == 19:
print("Aircraft address:", address)
print("Data:")
# print(bits_str)
print(bits_str[:5], bits_str[5:8], bits_str[8:10], bits_str[10:13], bits_str[13] ,bits_str[14:24], bits_str[24], bits_str[25:35], bits_str[35:36], bits_str[36:65])
subtype = int(bits_str[5:8], 2)
# https://mode-s.org/decode/adsb/airborne-velocity.html
spd, hdg, rocd = -1, -1, -1
if subtype == 1 or subtype == 2:
print("Velocity Subtype 1: Ground speed")
v_ew_sign = int(bits_str[13], 2)
v_ew = int(bits_str[14:24], 2) - 1       # east-west velocity
v_ns_sign = int(bits_str[24], 2)
v_ns = int(bits_str[25:35], 2) - 1       # north-south velocity
v_we = -1*v_ew if v_ew_sign else v_ew
v_sn = -1*v_ns if v_ns_sign else v_ns
spd = math.sqrt(v_sn*v_sn + v_we*v_we)  # unit in kts
hdg = math.atan2(v_we, v_sn)
hdg = math.degrees(hdg)                 # convert to degrees
hdg = hdg if hdg >= 0 else hdg + 360    # no negative val
if subtype == 3:
print("Subtype Subtype 3: Airspeed")
hdg = int(bits_str[14:24], 2)/1024.0*360.0
spd = int(bits_str[25:35], 2)
vr_sign = int(bits_str[36], 2)
vr = int(bits_str[36:45], 2)
rocd = -1*vr if vr_sign else vr         # rate of climb/descend
print("Speed (kts):", spd, "Rate:", rocd, "Heading:", hdg)
print()
# print()
def calc_coordinates():
def _cprN(lat, is_odd):
nl = _cprNL(lat) - is_odd
return nl if nl > 1 else 1
def _cprNL(lat):
try:
nz = 15
a = 1 - math.cos(math.pi / (2 * nz))
b = math.cos(math.pi / 180.0 * abs(lat)) ** 2
nl = 2 * math.pi / (math.acos(1 - a/b))
return int(math.floor(nl))
except:
# happens when latitude is +/-90 degree
return 1
def floor_(x):
return int(math.floor(x))
lat1b, lon1b, alt1b = "10111000111010011", "10000110111111000", "000101111001"
lat2b, lon2b, alt2b = "10010011101011100", "10000011000011011", "000101110111"
lat1, lon1, alt1 = int(lat1b, 2), int(lon1b, 2), int(alt1b, 2)
lat2, lon2, alt2 = int(lat2b, 2), int(lon2b, 2), int(alt2b, 2)
# 131072 is 2^17, since CPR lat and lon are 17 bits each
cprlat_even, cprlon_even = lat1/131072.0, lon1/131072.0
cprlat_odd, cprlon_odd = lat2/131072.0, lon2/131072.0
print(cprlat_even, cprlon_even)
j = floor_(59*cprlat_even - 60*cprlat_odd)
print(j)
air_d_lat_even = 360.0 / 60
air_d_lat_odd = 360.0 / 59
# Lat
lat_even = float(air_d_lat_even * (j % 60 + cprlat_even))
lat_odd = float(air_d_lat_odd * (j % 59 + cprlat_odd))
if lat_even >= 270:
lat_even = lat_even - 360
if lat_odd >= 270:
lat_odd = lat_odd - 360
# Lon
ni = _cprN(lat_even, 0)
m = floor_(cprlon_even * (_cprNL(lat_even)-1) - cprlon_odd * _cprNL(lat_even) + 0.5)
lon = (360.0 / ni) * (m % ni + cprlon_even)
print("Lat", lat_even, "Lon", lon)
# Altitude
# Q-bit (bit 48) indicates whether the altitude is encoded in multiples of 25 or 100 ft (0: 100 ft, 1: 25 ft)
# The value can represent altitudes from -1000 to +50175 ft.
if alt1b[-5] == '1':
bits = alt1b[:-5] + alt1b[-4:]
n = int(bits, 2)
alt_ft = n*25 - 1000
print("Alt (ft)", alt_ft)
fs, data = wavfile.read("adsb_20190311_191728Z_1090000kHz_RF.wav")
T = 1/fs
print("Sample rate %f MS/s" % (fs / 1e6))
print("Cnt samples %d" % len(data))
print("Duration: %f s" % (T * len(data)))
data = data.astype(float)
cnt = data.shape[0]
# Processing only part on file (faster):
# cnt = 10000000
# data = data[:cnt]
print("Processing I/Q...")
I, Q = data[:, 0], data[:, 1]
A = np.sqrt(I*I + Q*Q)
bits = np.zeros(cnt)
# To see scope without any processing, uncomment
# plt.plot(A)
# plt.show()
# sys.exit(0)
print("Extracting signals...")
pos = 0
avg = 200
msg_start = 0
# Find beginning of each signal
while pos < cnt - 16*1024:
# P1 - message start
while pos < cnt - 16*1024:
if A[pos] < avg and A[pos+1] > avg and pos - msg_start > 1000:
msg_start = pos
bits[pos] = 100
pos += 4
break
pos += 1
start1, start2, start3, start4 = msg_start, 0, 0, 0
# P2
while pos < cnt - 16*1024:
if A[pos] < avg and A[pos+1] > avg:
start2 = pos
bits[pos] = 90
pos += 1
break
pos += 1
# P3
while pos < cnt - 16*1024:
if A[pos] < avg and A[pos+1] > avg:
start3 = pos
bits[pos] = 80
pos += 1
break
pos += 1
# P4
while pos < cnt - 16*1024:
if A[pos] < avg and A[pos+1] > avg:
start4 = pos
bits[pos] = 70
pos += 1
break
pos += 1
sig_diff = start4 - start1
if 20 < sig_diff < 25:
bits[msg_start] = 500
bit_len = int((start4 - start1) / 4.5)
# print(pos, start1, start4, ' - ', bit_len)
# start = start1 + 8*bit_len
parse_message(A, msg_start, bit_len)
pos += 450
# For debugging: check signal start
# plt.plot(A)
# plt.plot(bits)
# plt.show()

Spero aliquem interesse, propter animum tuum.

Source: www.habr.com

Add a comment