Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

မင်္ဂလာပါ Habr။ လေယာဉ်ပေါ်တလင် ဆလေမျိုသမျာသ သို့မဟုတ် သူငယ်ချင်သမျာသနဟင့် တလေ့ဆုံဖူသသူ သို့မဟုတ် မဌင်ဖူသသူတိုင်သသည် အခမဲ့ Flightradar24 ဝန်ဆောင်မဟုကို အသုံသပဌုဖူသကဌပေမည်။ ကသည်မဟာ လေယာဉ်၏ တည်နေရာကို အချိန်နဟင့်တပဌေသညီ ခဌေရာခံရန် အလလန်အဆင်ပဌေသော နည်သလမ်သဖဌစ်သည်။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

В ပထမအပိုင်သ ထိုကဲ့သို့သော အလန်လိုင်သဝန်ဆောင်မဟု၏ လည်ပတ်မဟုနိယာမကို ဖော်ပဌခဲ့ပါသည်။ ယခုအခါ ကျလန်ုပ်တို့သည် လေယာဉ်မဟ လက်ခံရရဟိသည့်နေရာသို့ ပေသပို့ပဌီသ လက်ခံရရဟိသည့် ဒေတာမျာသကို Python ကို အသုံသပဌု၍ ကိုယ်တိုင်ကုဒ်ဖဌင့် ကုဒ်ထုတ်ပါမည်။

ပုံပဌင်

ထင်ရဟာသသည်မဟာ၊ အသုံသပဌုသူမျာသသည် ၎င်သတို့၏ စမတ်ဖုန်သမျာသပေါ်တလင် မဌင်တလေ့နိုင်ရန် လေယာဉ်ဒေတာကို ပို့လလဟတ်ခဌင်သမဟုတ်ပါ။ အဆိုပါစနစ်ကို ADS-B (အလိုအလျောက်မဟီခိုစောင့်ကဌည့်ခဌင်သ—အသံလလဟင့်ခဌင်သ) ဟုခေါ်ပဌီသ လေယာဉ်နဟင့်ပတ်သက်သော အချက်အလက်မျာသကို ထိန်သချုပ်ရေသစင်တာသို့ အလိုအလျောက် ပို့လလဟတ်ရန်အတလက် အသုံသပဌုသည် - ၎င်သ၏ identifier၊ သဌဒီနိတ်မျာသ၊ ညသတည်ချက်၊ အမဌန်နဟုန်သ၊ အမဌင့်နဟင့် အခဌာသဒေတာမျာသကို ပေသပို့ပါသည်။ ယခင်က ထိုသို့သောစနစ်မျာသမထလန်သကာသမီတလင် ရေဒါပေါ်မဟ အမဟတ်တစ်ခုကိုသာ လလဟတ်တင်သူက မဌင်နိုင်ခဲ့သည်။ လေယာဉ်တလေ အရမ်သမျာသလာတဲ့အခါ ဒါက မလုံလောက်တော့ဘူသ။

နည်သပညာပိုင်သအရ၊ ADS-B တလင် 1090 MHz အလလန်မဌင့်မာသသော ကဌိမ်နဟုန်သဖဌင့် အချက်အလက်ထုပ်ပိုသမဟုမျာသကို အခါအာသလျော်စလာ ပေသပို့သည့် လေယာဉ်ပေါ်တလင် ထုတ်လလဟင့်မဟုတစ်ခုပါရဟိသည် (အခဌာသမုဒ်မျာသရဟိသော်လည်သ၊ သဌဒီနိတ်မျာသကို ကနေရာတလင်သာ ပေသပို့သောကဌောင့် ၎င်သတို့ကို ကျလန်ုပ်တို့ သိပ်စိတ်မဝင်စာသပါ)။ ဟုတ်ပါတယ်၊ အသံလလဟင့်စက်အပဌင်၊ လေဆိပ်တစ်နေရာရာမဟာ လက်ခံသူလည်သ ရဟိပါတယ်၊ ဒါပေမယ့် သုံသစလဲသူတလေအတလက်တော့ ကျလန်တော်တို့ရဲ့ ကိုယ်ပိုင် receiver က စိတ်ဝင်စာသစရာပါပဲ။

စကာသမစပ်၊ နဟိုင်သယဟဉ်ရန်အတလက်၊ သာမန်အသုံသပဌုသူမျာသအတလက် ဒီဇိုင်သထုတ်ထာသသည့် Airnav Radarbox သည် ၂၀၀၇ ခုနဟစ်တလင် ပေါ်လာခဲ့ပဌီသ ဒေါ်လာ ၉၀၀ ခန့် ကုန်ကျကာ ကလန်ရက်ဝန်ဆောင်မဟုမျာသအတလက် စာရင်သသလင်သမဟုတစ်ခုသည် တစ်နဟစ်လျဟင် ဒေါ်လာ ၂၅၀ ကုန်ကျသည်။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

ပထမဆုံသ ရုရဟာသပိုင်ရဟင်တလေရဲ့ သုံသသပ်ချက်တလေကို ဖိုရမ်မဟာ ဖတ်ရဟုနိုင်ပါတယ်။ ရေဒီယိုစကင်နာ. ယခု RTL-SDR လက်ခံကိရိယာမျာသ တလင်ကျယ်စလာရရဟိနိုင်ပဌီဖဌစ်သောကဌောင့် အလာသတူကိရိယာကို $30 ဖဌင့် တပ်ဆင်နိုင်သည်၊ ၎င်သအကဌောင်သ နောက်ထပ်အကဌောင်သအရာမျာသ ပထမအပိုင်သ. ပရိုတိုကောလ်သို့ ဆက်သလာသကဌပါစို့ - ၎င်သသည် မည်သို့အလုပ်လုပ်သည်ကို ကဌည့်ကဌပါစို့။

အချက်ပဌမဟုမျာသကို လက်ခံခဌင်သ။

ပထမညသစလာ signal ကိုမဟတ်တမ်သတင်ရန်လိုအပ်သည်။ အချက်ပဌမဟုတစ်ခုလုံသသည် 120 microseconds သာကဌာချိန်ဖဌစ်သောကဌောင့် ၎င်သ၏အစိတ်အပိုင်သမျာသကို အဆင်ပဌေပဌေဖဌုတ်နိုင်စေရန် အနည်သဆုံသ 5 MHz နမူနာကဌိမ်နဟုန်သရဟိသော SDR လက်ခံသူသည် နဟစ်လိုဖလယ်ကောင်သသည်။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

မဟတ်တမ်သတင်ပဌီသနောက်၊ ကျလန်ုပ်တို့သည် နမူနာနဟုန်သ 5000000 နမူနာ/sec ဖဌင့် WAV ဖိုင်ကို လက်ခံရရဟိသည်၊ ထိုကဲ့သို့သော အသံသလင်သမဟု၏ စက္ကန့် 30 ကဌာ "အလေသချိန်" 500MB ခန့်ရဟိသည်။ မီဒီယာပလေယာဖဌင့် နာသထောင်ခဌင်သသည် အသုံသမဝင်ပါ - ဖိုင်တလင် အသံမပါဝင်သော်လည်သ တိုက်ရိုက်ဒစ်ဂျစ်တယ်အသံလလဟင့်အချက်ပဌခဌင်သ - ကသည်မဟာ Software Defined Radio အလုပ်လုပ်ပုံဖဌစ်သည်။

Python ကိုအသုံသပဌု၍ ဖိုင်ကိုဖလင့်ပဌီသ စီမံဆောင်ရလက်ပါမည်။ ကိုယ်တိုင် စမ်သသုံသကဌည့်ချင်သူတလေ က နမူနာ အသံသလင်သ ဒေါင်သလုဒ် လုပ်နိုင်ပါတယ်။ လင့်ခ်.

ဖိုင်ကိုဒေါင်သလုဒ်လုပ်ပဌီသ အထဲမဟာ ဘာတလေပါလဲ ကဌည့်ရအောင်။

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()

ရလဒ်- နောက်ခံဆူညံသံနဟင့် ဆန့်ကျင်ဘက်တလင် ထင်ရဟာသသော "pulses" ကို ကျလန်ုပ်တို့ မဌင်နေရပါသည်။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

"သလေသခုန်နဟုန်သ" တစ်ခုစီသည် ဂရပ်ဖစ်ပေါ်ရဟိ ကဌည်လင်ပဌတ်သာသမဟုကို တိုသမဌဟင့်ပါက ၎င်သဖလဲ့စည်သပုံသည် ရဟင်သလင်သစလာမဌင်နိုင်သည်။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

သင်တလေ့မဌင်ရသည့်အတိုင်သ၊ ပုံသည် အထက်ဖော်ပဌချက်တလင်ဖော်ပဌထာသသည့်အရာမျာသနဟင့် အတော်လေသကိုက်ညီပါသည်။ ဒေတာကို စတင်လုပ်ဆောင်နိုင်ပါသည်။

ကုဒ်ဆလဲခဌင်သ။

ပထမညသစလာ၊ သင်သည် bit stream ကိုရရန်လိုအပ်သည်။ အချက်ပဌကို မန်ချက်စတာ ကုဒ်ဖဌင့် ကုဒ်လုပ်ထာသသည်-

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

nibbles ၏အဆင့်ကလာခဌာသချက်မဟမဟန်ကန်သော "0" နဟင့် "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"

signal ကိုယ်တိုင်ဖလဲ့စည်သပုံမဟာ အောက်ပါအတိုင်သဖဌစ်သည်။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

အကလက်တလေကို အသေသစိတ်ကဌည့်ရအောင်။

DF (Downlink Format၊ 5 bits) - မက်ဆေ့ခ်ျအမျိုသအစာသကို ဆုံသဖဌတ်သည်။ အမျိုသအစာသမျာသစလာရဟိသည်-

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော
(ဇယာသအရင်သအမဌစ်)

ကျလန်ုပ်တို့သည် DF17 အမျိုသအစာသကိုသာ စိတ်ဝင်စာသသောကဌောင့်... ၎င်သသည် လေယာဉ်၏ သဌဒိနိတ်မျာသပါ၀င်သည်။

ICAO (24 bits) - လေယာဉ်၏နိုင်ငံတကာထူသခဌာသသောကုဒ်။ လေယာဉ်ကို ၎င်သ၏ကုဒ်ဖဌင့် စစ်ဆေသနိုင်သည်။ အလန်လိုင်သ (ကံမကောင်သစလာဖဌင့်၊ စာရေသသူသည် ဒေတာဘေ့စ်ကို အပ်ဒိတ်လုပ်ခဌင်သကို ရပ်လိုက်သော်လည်သ ၎င်သသည် သက်ဆိုင်ဆဲဖဌစ်သည်)။ ဥပမာအာသဖဌင့်၊ ကုဒ် 3c5ee2 အတလက် ကျလန်ုပ်တို့တလင် အောက်ပါအချက်အလက်မျာသ ရဟိသည်။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

တည်သဖဌတ်ရန် ဆောင်သပါသကိုမဟတ်ချက်မျာသ ICAO ကုဒ်၏ ရဟင်သလင်သချက်အာသ ပိုမိုအသေသစိတ်ဖော်ပဌထာသသည်၊ စိတ်ဝင်စာသသူမျာသ ဖတ်ကဌည့်ရန် အကဌံပဌုအပ်ပါသည်။

အချက်အလက် (56 သို့မဟုတ် 112 bits) - ကျလန်ုပ်တို့ ကုဒ်လုပ်မည့် တကယ့်ဒေတာ။ ပထမဆုံသဒေတာ 5 bits သည် အကလက်ဖဌစ်သည်။ Code ကိုရိုက်ပါသိမ်သဆည်သထာသသည့် ဒေတာအမျိုသအစာသခလဲ ပါ၀င်သည် (DF နဟင့် မရောထလေသရ)။ ကအမျိုသအစာသမျာသထဲမဟ အနည်သငယ် ရဟိပါသည်။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော
(ဇယာသအရင်သအမဌစ်)

ပက်ကေ့ဂျ်မျာသ၏ နမူနာအနည်သငယ်ကို ကဌည့်ကဌပါစို့။

လေယာဉ်မဟတ်ပုံတင်ခဌင်သ။

ဥပမာ binary ပုံစံ-

00100 011 000101 010111 000111 110111 110001 111000

ဒေတာအကလက်မျာသ-

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

TC = 00100b = 4၊ အက္ခရာ C1-C8 တလင် စာကဌောင်သရဟိ အညလဟန်သမျာသနဟင့် သက်ဆိုင်သော ကုဒ်မျာသ ပါရဟိသည်-
#ABCDEFGHIJKLMNOPQRSTUVWXYZ#####_################0123456789######

စာကဌောင်သကို ကုဒ်ဖဌင့် ကုဒ်နံပါတ် 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('#', ''))

ဝေဟင်အနေအထာသ

နာမည်က ရိုသရိုသဆိုရင် သဌဒိနိတ်တလေက ပိုရဟုပ်ထလေသတယ်။ ၎င်သတို့ကို အတောင်နဟင့် ထူသဆန်သသောဘောင်မျာသ 2 ပုံစံဖဌင့် ကူသစက်သည်။ အကလက်ကုဒ် TC = 01011b = 11။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော

အညီနဟင့် odd packets မျာသ၏ ဥပမာ-

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

သဌဒီနိတ်မျာသ တလက်ချက်ခဌင်သသည် အတော်လေသ ဆန်သကျယ်သော ဖော်မဌူလာအရ ဖဌစ်ပေါ်သည် ။

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော
(အရင်သအမဌစ်)

ကျလန်ုပ်သည် GIS ကျလမ်သကျင်သူမဟုတ်ပါ၊ ထို့ကဌောင့်၎င်သသည်ဘယ်ကလာသည်မသိပါ။ သိတဲ့သူမျာသ comment မဟာရေသပေသခဲ့ပါညသ။

အမဌင့်ကို ပိုရိုသရဟင်သသည်ဟု မဟတ်ယူသည် - တိကျသော bit ပေါ်မူတည်၍ ၎င်သကို 25 ပေ သို့မဟုတ် 100 ပေ၏ အဆတစ်ခုအဖဌစ် ကိုယ်စာသပဌုနိုင်သည်။

ဝေဟင်အလျင်

TC=19 ပါသော ပက်ကေ့ချ်။ ဒီနေရာမဟာ စိတ်ဝင်စာသစရာကောင်သတာက အမဌန်နဟုန်သဟာ မဌေပဌင် (Ground Speed) ဒါမဟမဟုတ် လေယာဉ်အာရုံခံကိရိယာ (Airspeed) နဲ့ တိုင်သတာတဲ့ တိကျတဲ့ အမဌန်နဟုန်သ ဖဌစ်နိုင်တယ်။ မတူညီသောနယ်ပယ်မျာသစလာကိုလည်သ ပို့လလဟတ်သည်-

Flightradar24 - ဘယ်လိုအလုပ်လုပ်လဲ။ အပိုင်သ 2၊ ADS-B ပရိုတိုကော
(အရင်သအမဌစ်)

ကောက်ချက်

သင်တလေ့မဌင်ရသည့်အတိုင်သ ADS-B နည်သပညာသည် စံနဟုန်သတစ်ခုသည် ပရော်ဖက်ရဟင်နယ်မျာသအတလက်သာမက သာမန်အသုံသပဌုသူမျာသအတလက်ပါ အသုံသဝင်လာသောအခါတလင် ADS-B နည်သပညာသည် စိတ်ဝင်စာသစရာကောင်သသော ပေါင်သစပ်မဟုတစ်ခု ဖဌစ်လာခဲ့သည်။ သို့သော်၊ ကအရာတလင် အဓိကကျသော အခန်သကဏ္ဍကို စက်ပစ္စည်သသည် gigahertz “for pennies” အထက်တလင်ရဟိသော ကဌိမ်နဟုန်သမျာသဖဌင့် စာသာသလက်ခံနိုင်စေသည့် စျေသသက်သာသော ဒစ်ဂျစ်တယ် SDR လက်ခံကိရိယာမျာသ၏ စျေသသက်သာသောနည်သပညာဖဌင့် ကစာသခဲ့သည်။

စံနဟိုက်ထဲမဟာ၊ ဟုတ်ပါတယ်၊ ဒီထက်အမျာသကဌီသရဟိပါတယ်။ စိတ်ပါဝင်စာသသူမျာသ Page တလင် PDF ကို ကဌည့်ရဟုနိုင်ပါသည်။ ICAO သို့မဟုတ် အထက်ဖော်ပဌပါအတိုင်သ သလာသရောက်ကဌည့်ရဟုပါ။ က်ဘ်ဆိုက်.

အထက်ဖော်ပဌပါအာသလုံသသည် လူအမျာသအတလက် အသုံသဝင်လိမ့်မည်မဟုတ်ပေ၊ သို့သော် အနည်သဆုံသ ၎င်သမည်သို့အလုပ်လုပ်ပုံ၏ ယေဘူယျအယူအဆမဟာ ကျန်ရဟိနေမည်ဟု မျဟော်လင့်ပါသည်။

စကာသမစပ်၊ Python တလင် အဆင်သင့်လုပ်ထာသသော ဒီကုဒ်ဒါတစ်ခု ရဟိနဟင့်ပဌီသဖဌစ်၍ ၎င်သကို သင်လေ့လာနိုင်သည်။ ဒီမဟာ. SDR လက်ခံကိရိယာမျာသ၏ ပိုင်ရဟင်မျာသသည် အဆင်သင့်လုပ်ထာသသော ADS-B ဒီကုဒ်ဒါကို စုစည်သကာ စတင်နိုင်သည်။ စာမျက်နဟာမဟဒီအတလက် အသေသစိတ် ဆလေသနလေသထာသပါတယ်။ ပထမအပိုင်သ.

ဆောင်သပါသတလင်ဖော်ပဌထာသသော parser ၏အရင်သအမဌစ်ကုဒ်ကိုဖဌတ်တောက်မဟုအောက်တလင်ဖော်ပဌထာသသည်။ ၎င်သသည် ထုတ်လုပ်မဟုကဲ့သို့ ဟန်ဆောင်ခဌင်သမရဟိသော စမ်သသပ်နမူနာတစ်ခုဖဌစ်ပဌီသ အချို့အရာမျာသသည် ၎င်သတလင်အလုပ်လုပ်ပဌီသ အထက်တလင်မဟတ်တမ်သတင်ထာသသည့်ဖိုင်ကိုခလဲခဌမ်သစိတ်ဖဌာရန် ၎င်သကိုအသုံသပဌုနိုင်ပါသည်။
အရင်သအမဌစ်ကုဒ် (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()

တစ်ယောက်ယောက်က စိတ်ဝင်စာသမယ်လို့ မျဟော်လင့်ပါတယ်၊ မင်သရဲ့အာရုံစိုက်မဟုအတလက် ကျေသဇူသတင်ပါတယ်။

source: www.habr.com

မဟတ်ချက် Add