Wideband autonomous SDR analysis engine forked from sdr-visual-suite
25'ten fazla konu seçemezsiniz Konular bir harf veya rakamla başlamalı, kısa çizgiler ('-') içerebilir ve en fazla 35 karakter uzunluğunda olabilir.

449 satır
16KB

  1. package config
  2. import (
  3. "math"
  4. "os"
  5. "time"
  6. "gopkg.in/yaml.v3"
  7. )
  8. type Band struct {
  9. Name string `yaml:"name" json:"name"`
  10. StartHz float64 `yaml:"start_hz" json:"start_hz"`
  11. EndHz float64 `yaml:"end_hz" json:"end_hz"`
  12. }
  13. type DetectorConfig struct {
  14. ThresholdDb float64 `yaml:"threshold_db" json:"threshold_db"`
  15. MinDurationMs int `yaml:"min_duration_ms" json:"min_duration_ms"`
  16. HoldMs int `yaml:"hold_ms" json:"hold_ms"`
  17. EmaAlpha float64 `yaml:"ema_alpha" json:"ema_alpha"`
  18. HysteresisDb float64 `yaml:"hysteresis_db" json:"hysteresis_db"`
  19. MinStableFrames int `yaml:"min_stable_frames" json:"min_stable_frames"`
  20. GapToleranceMs int `yaml:"gap_tolerance_ms" json:"gap_tolerance_ms"`
  21. CFARMode string `yaml:"cfar_mode" json:"cfar_mode"`
  22. CFARGuardHz float64 `yaml:"cfar_guard_hz" json:"cfar_guard_hz"`
  23. CFARTrainHz float64 `yaml:"cfar_train_hz" json:"cfar_train_hz"`
  24. CFARGuardCells int `yaml:"cfar_guard_cells,omitempty" json:"cfar_guard_cells,omitempty"`
  25. CFARTrainCells int `yaml:"cfar_train_cells,omitempty" json:"cfar_train_cells,omitempty"`
  26. CFARRank int `yaml:"cfar_rank" json:"cfar_rank"`
  27. CFARScaleDb float64 `yaml:"cfar_scale_db" json:"cfar_scale_db"`
  28. CFARWrapAround bool `yaml:"cfar_wrap_around" json:"cfar_wrap_around"`
  29. EdgeMarginDb float64 `yaml:"edge_margin_db" json:"edge_margin_db"`
  30. MaxSignalBwHz float64 `yaml:"max_signal_bw_hz" json:"max_signal_bw_hz"`
  31. MergeGapHz float64 `yaml:"merge_gap_hz" json:"merge_gap_hz"`
  32. ClassHistorySize int `yaml:"class_history_size" json:"class_history_size"`
  33. ClassSwitchRatio float64 `yaml:"class_switch_ratio" json:"class_switch_ratio"`
  34. // Deprecated (backward compatibility)
  35. CFAREnabled *bool `yaml:"cfar_enabled,omitempty" json:"cfar_enabled,omitempty"`
  36. }
  37. type RecorderConfig struct {
  38. Enabled bool `yaml:"enabled" json:"enabled"`
  39. MinSNRDb float64 `yaml:"min_snr_db" json:"min_snr_db"`
  40. MinDuration string `yaml:"min_duration" json:"min_duration"`
  41. MaxDuration string `yaml:"max_duration" json:"max_duration"`
  42. PrerollMs int `yaml:"preroll_ms" json:"preroll_ms"`
  43. RecordIQ bool `yaml:"record_iq" json:"record_iq"`
  44. RecordAudio bool `yaml:"record_audio" json:"record_audio"`
  45. AutoDemod bool `yaml:"auto_demod" json:"auto_demod"`
  46. AutoDecode bool `yaml:"auto_decode" json:"auto_decode"`
  47. MaxDiskMB int `yaml:"max_disk_mb" json:"max_disk_mb"`
  48. OutputDir string `yaml:"output_dir" json:"output_dir"`
  49. ClassFilter []string `yaml:"class_filter" json:"class_filter"`
  50. RingSeconds int `yaml:"ring_seconds" json:"ring_seconds"`
  51. // Audio quality settings (AQ-2, AQ-3, AQ-5)
  52. DeemphasisUs float64 `yaml:"deemphasis_us" json:"deemphasis_us"` // De-emphasis time constant in µs. 50=Europe, 75=US/Japan, 0=disabled. Default: 50
  53. ExtractionTaps int `yaml:"extraction_fir_taps" json:"extraction_fir_taps"` // FIR tap count for extraction filter. Default: 101, max 301
  54. ExtractionBwMult float64 `yaml:"extraction_bw_mult" json:"extraction_bw_mult"` // BW multiplier for extraction. Default: 1.2 (20% wider than detected)
  55. }
  56. type DecoderConfig struct {
  57. FT8Cmd string `yaml:"ft8_cmd" json:"ft8_cmd"`
  58. WSPRCmd string `yaml:"wspr_cmd" json:"wspr_cmd"`
  59. DMRCmd string `yaml:"dmr_cmd" json:"dmr_cmd"`
  60. DStarCmd string `yaml:"dstar_cmd" json:"dstar_cmd"`
  61. FSKCmd string `yaml:"fsk_cmd" json:"fsk_cmd"`
  62. PSKCmd string `yaml:"psk_cmd" json:"psk_cmd"`
  63. }
  64. type PipelineGoalConfig struct {
  65. Intent string `yaml:"intent" json:"intent"`
  66. MonitorStartHz float64 `yaml:"monitor_start_hz" json:"monitor_start_hz"`
  67. MonitorEndHz float64 `yaml:"monitor_end_hz" json:"monitor_end_hz"`
  68. MonitorSpanHz float64 `yaml:"monitor_span_hz" json:"monitor_span_hz"`
  69. SignalPriorities []string `yaml:"signal_priorities" json:"signal_priorities"`
  70. AutoRecordClasses []string `yaml:"auto_record_classes" json:"auto_record_classes"`
  71. AutoDecodeClasses []string `yaml:"auto_decode_classes" json:"auto_decode_classes"`
  72. }
  73. type PipelineConfig struct {
  74. Mode string `yaml:"mode" json:"mode"`
  75. Goals PipelineGoalConfig `yaml:"goals" json:"goals"`
  76. }
  77. type SurveillanceConfig struct {
  78. AnalysisFFTSize int `yaml:"analysis_fft_size" json:"analysis_fft_size"`
  79. FrameRate int `yaml:"frame_rate" json:"frame_rate"`
  80. Strategy string `yaml:"strategy" json:"strategy"`
  81. DisplayBins int `yaml:"display_bins" json:"display_bins"`
  82. DisplayFPS int `yaml:"display_fps" json:"display_fps"`
  83. }
  84. type RefinementConfig struct {
  85. Enabled bool `yaml:"enabled" json:"enabled"`
  86. MaxConcurrent int `yaml:"max_concurrent" json:"max_concurrent"`
  87. MinCandidateSNRDb float64 `yaml:"min_candidate_snr_db" json:"min_candidate_snr_db"`
  88. MinSpanHz float64 `yaml:"min_span_hz" json:"min_span_hz"`
  89. MaxSpanHz float64 `yaml:"max_span_hz" json:"max_span_hz"`
  90. AutoSpan *bool `yaml:"auto_span" json:"auto_span"`
  91. }
  92. type ResourceConfig struct {
  93. PreferGPU bool `yaml:"prefer_gpu" json:"prefer_gpu"`
  94. MaxRefinementJobs int `yaml:"max_refinement_jobs" json:"max_refinement_jobs"`
  95. MaxRecordingStreams int `yaml:"max_recording_streams" json:"max_recording_streams"`
  96. MaxDecodeJobs int `yaml:"max_decode_jobs" json:"max_decode_jobs"`
  97. }
  98. type ProfileConfig struct {
  99. Name string `yaml:"name" json:"name"`
  100. Description string `yaml:"description" json:"description"`
  101. Pipeline *PipelineConfig `yaml:"pipeline,omitempty" json:"pipeline,omitempty"`
  102. Surveillance *SurveillanceConfig `yaml:"surveillance,omitempty" json:"surveillance,omitempty"`
  103. Refinement *RefinementConfig `yaml:"refinement,omitempty" json:"refinement,omitempty"`
  104. Resources *ResourceConfig `yaml:"resources,omitempty" json:"resources,omitempty"`
  105. }
  106. type Config struct {
  107. Bands []Band `yaml:"bands" json:"bands"`
  108. CenterHz float64 `yaml:"center_hz" json:"center_hz"`
  109. SampleRate int `yaml:"sample_rate" json:"sample_rate"`
  110. FFTSize int `yaml:"fft_size" json:"fft_size"`
  111. GainDb float64 `yaml:"gain_db" json:"gain_db"`
  112. TunerBwKHz int `yaml:"tuner_bw_khz" json:"tuner_bw_khz"`
  113. UseGPUFFT bool `yaml:"use_gpu_fft" json:"use_gpu_fft"`
  114. ClassifierMode string `yaml:"classifier_mode" json:"classifier_mode"`
  115. AGC bool `yaml:"agc" json:"agc"`
  116. DCBlock bool `yaml:"dc_block" json:"dc_block"`
  117. IQBalance bool `yaml:"iq_balance" json:"iq_balance"`
  118. Pipeline PipelineConfig `yaml:"pipeline" json:"pipeline"`
  119. Surveillance SurveillanceConfig `yaml:"surveillance" json:"surveillance"`
  120. Refinement RefinementConfig `yaml:"refinement" json:"refinement"`
  121. Resources ResourceConfig `yaml:"resources" json:"resources"`
  122. Profiles []ProfileConfig `yaml:"profiles" json:"profiles"`
  123. Detector DetectorConfig `yaml:"detector" json:"detector"`
  124. Recorder RecorderConfig `yaml:"recorder" json:"recorder"`
  125. Decoder DecoderConfig `yaml:"decoder" json:"decoder"`
  126. WebAddr string `yaml:"web_addr" json:"web_addr"`
  127. EventPath string `yaml:"event_path" json:"event_path"`
  128. FrameRate int `yaml:"frame_rate" json:"frame_rate"`
  129. WaterfallLines int `yaml:"waterfall_lines" json:"waterfall_lines"`
  130. WebRoot string `yaml:"web_root" json:"web_root"`
  131. }
  132. func Default() Config {
  133. return Config{
  134. Bands: []Band{
  135. {Name: "example", StartHz: 99.5e6, EndHz: 100.5e6},
  136. },
  137. CenterHz: 100.0e6,
  138. SampleRate: 2_048_000,
  139. FFTSize: 2048,
  140. GainDb: 30,
  141. TunerBwKHz: 1536,
  142. UseGPUFFT: false,
  143. ClassifierMode: "combined",
  144. AGC: false,
  145. DCBlock: false,
  146. IQBalance: false,
  147. Pipeline: PipelineConfig{
  148. Mode: "legacy",
  149. Goals: PipelineGoalConfig{
  150. Intent: "general-monitoring",
  151. },
  152. },
  153. Surveillance: SurveillanceConfig{
  154. AnalysisFFTSize: 2048,
  155. FrameRate: 15,
  156. Strategy: "single-resolution",
  157. DisplayBins: 2048,
  158. DisplayFPS: 15,
  159. },
  160. Refinement: RefinementConfig{
  161. Enabled: true,
  162. MaxConcurrent: 8,
  163. MinCandidateSNRDb: 0,
  164. MinSpanHz: 0,
  165. MaxSpanHz: 0,
  166. AutoSpan: boolPtr(true),
  167. },
  168. Resources: ResourceConfig{
  169. PreferGPU: true,
  170. MaxRefinementJobs: 8,
  171. MaxRecordingStreams: 16,
  172. MaxDecodeJobs: 16,
  173. },
  174. Profiles: []ProfileConfig{
  175. {Name: "legacy", Description: "Current single-band pipeline behavior", Pipeline: &PipelineConfig{Mode: "legacy", Goals: PipelineGoalConfig{Intent: "general-monitoring"}}},
  176. {Name: "wideband-balanced", Description: "Prepared baseline for scalable wideband surveillance", Pipeline: &PipelineConfig{Mode: "wideband-balanced", Goals: PipelineGoalConfig{Intent: "wideband-surveillance"}}},
  177. {Name: "wideband-aggressive", Description: "Higher surveillance/refinement budgets for future broad-span monitoring", Pipeline: &PipelineConfig{Mode: "wideband-aggressive", Goals: PipelineGoalConfig{Intent: "high-density-wideband-surveillance"}}},
  178. {Name: "archive", Description: "Record-first monitoring profile", Pipeline: &PipelineConfig{Mode: "archive", Goals: PipelineGoalConfig{Intent: "archive-and-triage"}}},
  179. },
  180. Detector: DetectorConfig{
  181. ThresholdDb: -20,
  182. MinDurationMs: 250,
  183. HoldMs: 500,
  184. EmaAlpha: 0.2,
  185. HysteresisDb: 3,
  186. MinStableFrames: 3,
  187. GapToleranceMs: 500,
  188. CFARMode: "GOSCA",
  189. CFARGuardHz: 500,
  190. CFARTrainHz: 5000,
  191. CFARGuardCells: 3,
  192. CFARTrainCells: 24,
  193. CFARRank: 36,
  194. CFARScaleDb: 6,
  195. CFARWrapAround: true,
  196. EdgeMarginDb: 3.0,
  197. MaxSignalBwHz: 150000,
  198. MergeGapHz: 5000,
  199. ClassHistorySize: 10,
  200. ClassSwitchRatio: 0.6,
  201. },
  202. Recorder: RecorderConfig{
  203. Enabled: false,
  204. MinSNRDb: 10,
  205. MinDuration: "1s",
  206. MaxDuration: "300s",
  207. PrerollMs: 500,
  208. RecordIQ: true,
  209. RecordAudio: false,
  210. AutoDemod: true,
  211. AutoDecode: false,
  212. MaxDiskMB: 0,
  213. OutputDir: "data/recordings",
  214. RingSeconds: 8,
  215. DeemphasisUs: 50,
  216. ExtractionTaps: 101,
  217. ExtractionBwMult: 1.2,
  218. },
  219. Decoder: DecoderConfig{},
  220. WebAddr: ":8080",
  221. EventPath: "data/events.jsonl",
  222. FrameRate: 15,
  223. WaterfallLines: 200,
  224. WebRoot: "web",
  225. }
  226. }
  227. func Load(path string) (Config, error) {
  228. cfg := Default()
  229. if b, err := os.ReadFile(autosavePath(path)); err == nil {
  230. if err := yaml.Unmarshal(b, &cfg); err == nil {
  231. return applyDefaults(cfg), nil
  232. }
  233. }
  234. b, err := os.ReadFile(path)
  235. if err != nil {
  236. return cfg, err
  237. }
  238. if err := yaml.Unmarshal(b, &cfg); err != nil {
  239. return cfg, err
  240. }
  241. return applyDefaults(cfg), nil
  242. }
  243. func applyDefaults(cfg Config) Config {
  244. if cfg.Detector.MinDurationMs <= 0 {
  245. cfg.Detector.MinDurationMs = 250
  246. }
  247. if cfg.Detector.HoldMs <= 0 {
  248. cfg.Detector.HoldMs = 500
  249. }
  250. if cfg.Detector.MinStableFrames <= 0 {
  251. cfg.Detector.MinStableFrames = 3
  252. }
  253. if cfg.Detector.GapToleranceMs <= 0 {
  254. cfg.Detector.GapToleranceMs = cfg.Detector.HoldMs
  255. }
  256. if cfg.Detector.CFARMode == "" {
  257. if cfg.Detector.CFAREnabled != nil {
  258. if *cfg.Detector.CFAREnabled {
  259. cfg.Detector.CFARMode = "OS"
  260. } else {
  261. cfg.Detector.CFARMode = "OFF"
  262. }
  263. } else {
  264. cfg.Detector.CFARMode = "GOSCA"
  265. }
  266. }
  267. if cfg.Detector.CFARGuardHz <= 0 && cfg.Detector.CFARGuardCells > 0 {
  268. cfg.Detector.CFARGuardHz = float64(cfg.Detector.CFARGuardCells) * 62.5
  269. }
  270. if cfg.Detector.CFARTrainHz <= 0 && cfg.Detector.CFARTrainCells > 0 {
  271. cfg.Detector.CFARTrainHz = float64(cfg.Detector.CFARTrainCells) * 62.5
  272. }
  273. if cfg.Detector.CFARGuardHz <= 0 {
  274. cfg.Detector.CFARGuardHz = 500
  275. }
  276. if cfg.Detector.CFARTrainHz <= 0 {
  277. cfg.Detector.CFARTrainHz = 5000
  278. }
  279. if cfg.Detector.CFARGuardCells <= 0 {
  280. cfg.Detector.CFARGuardCells = 3
  281. }
  282. if cfg.Detector.CFARTrainCells <= 0 {
  283. cfg.Detector.CFARTrainCells = 24
  284. }
  285. if cfg.Detector.CFARRank <= 0 || cfg.Detector.CFARRank > 2*cfg.Detector.CFARTrainCells {
  286. cfg.Detector.CFARRank = int(math.Round(0.75 * float64(2*cfg.Detector.CFARTrainCells)))
  287. if cfg.Detector.CFARRank <= 0 {
  288. cfg.Detector.CFARRank = 1
  289. }
  290. }
  291. if cfg.Detector.CFARScaleDb <= 0 {
  292. cfg.Detector.CFARScaleDb = 6
  293. }
  294. if cfg.Detector.EdgeMarginDb <= 0 {
  295. cfg.Detector.EdgeMarginDb = 3.0
  296. }
  297. if cfg.Detector.MaxSignalBwHz <= 0 {
  298. cfg.Detector.MaxSignalBwHz = 150000
  299. }
  300. if cfg.Detector.MergeGapHz <= 0 {
  301. cfg.Detector.MergeGapHz = 5000
  302. }
  303. if cfg.Detector.ClassHistorySize <= 0 {
  304. cfg.Detector.ClassHistorySize = 10
  305. }
  306. if cfg.Detector.ClassSwitchRatio <= 0 || cfg.Detector.ClassSwitchRatio > 1 {
  307. cfg.Detector.ClassSwitchRatio = 0.6
  308. }
  309. if cfg.Pipeline.Mode == "" {
  310. cfg.Pipeline.Mode = "legacy"
  311. }
  312. if cfg.Pipeline.Goals.Intent == "" {
  313. cfg.Pipeline.Goals.Intent = "general-monitoring"
  314. }
  315. if cfg.Pipeline.Goals.MonitorSpanHz <= 0 && cfg.Pipeline.Goals.MonitorStartHz != 0 && cfg.Pipeline.Goals.MonitorEndHz != 0 && cfg.Pipeline.Goals.MonitorEndHz > cfg.Pipeline.Goals.MonitorStartHz {
  316. cfg.Pipeline.Goals.MonitorSpanHz = cfg.Pipeline.Goals.MonitorEndHz - cfg.Pipeline.Goals.MonitorStartHz
  317. }
  318. if cfg.Surveillance.AnalysisFFTSize <= 0 {
  319. cfg.Surveillance.AnalysisFFTSize = cfg.FFTSize
  320. }
  321. if cfg.Surveillance.FrameRate <= 0 {
  322. cfg.Surveillance.FrameRate = cfg.FrameRate
  323. }
  324. if cfg.Surveillance.Strategy == "" {
  325. cfg.Surveillance.Strategy = "single-resolution"
  326. }
  327. if cfg.Surveillance.DisplayBins <= 0 {
  328. cfg.Surveillance.DisplayBins = cfg.FFTSize
  329. }
  330. if cfg.Surveillance.DisplayFPS <= 0 {
  331. cfg.Surveillance.DisplayFPS = cfg.FrameRate
  332. }
  333. if !cfg.Refinement.Enabled {
  334. // keep explicit false if user disabled it; enable by default only when unset-like zero config
  335. if cfg.Refinement.MaxConcurrent == 0 && cfg.Refinement.MinCandidateSNRDb == 0 {
  336. cfg.Refinement.Enabled = true
  337. }
  338. }
  339. if cfg.Refinement.MaxConcurrent <= 0 {
  340. cfg.Refinement.MaxConcurrent = 8
  341. }
  342. if cfg.Refinement.MinSpanHz < 0 {
  343. cfg.Refinement.MinSpanHz = 0
  344. }
  345. if cfg.Refinement.MaxSpanHz < 0 {
  346. cfg.Refinement.MaxSpanHz = 0
  347. }
  348. if cfg.Refinement.MaxSpanHz > 0 && cfg.Refinement.MinSpanHz > cfg.Refinement.MaxSpanHz {
  349. cfg.Refinement.MaxSpanHz = cfg.Refinement.MinSpanHz
  350. }
  351. if cfg.Refinement.AutoSpan == nil {
  352. cfg.Refinement.AutoSpan = boolPtr(true)
  353. }
  354. if cfg.Resources.MaxRefinementJobs <= 0 {
  355. cfg.Resources.MaxRefinementJobs = cfg.Refinement.MaxConcurrent
  356. }
  357. if cfg.Resources.MaxRecordingStreams <= 0 {
  358. cfg.Resources.MaxRecordingStreams = 16
  359. }
  360. if cfg.FrameRate <= 0 {
  361. cfg.FrameRate = 15
  362. }
  363. if cfg.WaterfallLines <= 0 {
  364. cfg.WaterfallLines = 200
  365. }
  366. if cfg.WebRoot == "" {
  367. cfg.WebRoot = "web"
  368. }
  369. if cfg.WebAddr == "" {
  370. cfg.WebAddr = ":8080"
  371. }
  372. if cfg.EventPath == "" {
  373. cfg.EventPath = "data/events.jsonl"
  374. }
  375. if cfg.SampleRate <= 0 {
  376. cfg.SampleRate = 2_048_000
  377. }
  378. if cfg.ClassifierMode == "" {
  379. cfg.ClassifierMode = "combined"
  380. }
  381. switch cfg.ClassifierMode {
  382. case "rule", "math", "combined":
  383. default:
  384. cfg.ClassifierMode = "combined"
  385. }
  386. if cfg.FFTSize <= 0 {
  387. cfg.FFTSize = 2048
  388. }
  389. if cfg.Surveillance.AnalysisFFTSize > 0 {
  390. cfg.FFTSize = cfg.Surveillance.AnalysisFFTSize
  391. } else {
  392. cfg.Surveillance.AnalysisFFTSize = cfg.FFTSize
  393. }
  394. if cfg.TunerBwKHz <= 0 {
  395. cfg.TunerBwKHz = 1536
  396. }
  397. if cfg.CenterHz == 0 {
  398. cfg.CenterHz = 100.0e6
  399. }
  400. if cfg.Recorder.OutputDir == "" {
  401. cfg.Recorder.OutputDir = "data/recordings"
  402. }
  403. if cfg.Recorder.RingSeconds <= 0 {
  404. cfg.Recorder.RingSeconds = 8
  405. }
  406. if cfg.Recorder.DeemphasisUs == 0 {
  407. cfg.Recorder.DeemphasisUs = 50
  408. }
  409. if cfg.Recorder.ExtractionTaps <= 0 {
  410. cfg.Recorder.ExtractionTaps = 101
  411. }
  412. if cfg.Recorder.ExtractionTaps > 301 {
  413. cfg.Recorder.ExtractionTaps = 301
  414. }
  415. if cfg.Recorder.ExtractionTaps%2 == 0 {
  416. cfg.Recorder.ExtractionTaps++ // must be odd
  417. }
  418. if cfg.Recorder.ExtractionBwMult <= 0 {
  419. cfg.Recorder.ExtractionBwMult = 1.2
  420. }
  421. return cfg
  422. }
  423. func (c Config) FrameInterval() time.Duration {
  424. fps := c.FrameRate
  425. if fps <= 0 {
  426. fps = 15
  427. }
  428. return time.Second / time.Duration(fps)
  429. }