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
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
use {
    std::{
        fmt,
        mem,
    },
};


pub use {
    fe_caps::*,
    fe_type::*,
    fe_sec_voltage::*,
    fe_sec_tone_mode::*,
    fe_sec_mini_cmd::*,
    fe_status::*,
    fe_spectral_inversion::*,
    fe_code_rate::*,
    fe_modulation::*,
    fe_transmit_mode::*,
    fe_guard_interval::*,
    fe_hierarchy::*,
    fe_interleaving::*,
    fe_pilot::*,
    fe_rolloff::*,
    fe_delivery_system::*,
    fecap_scale_params::*,
    dtv_property_cmd::*,
};


/// Frontend capabilities
mod fe_caps {
    /// There's something wrong at the frontend, and it can't report its capabilities
    pub const FE_IS_STUPID: u32                     = 0;
    /// Can auto-detect frequency spectral band inversion
    pub const FE_CAN_INVERSION_AUTO: u32            = 0x1;
    /// Supports FEC 1/2
    pub const FE_CAN_FEC_1_2: u32                   = 0x2;
    /// Supports FEC 2/3
    pub const FE_CAN_FEC_2_3: u32                   = 0x4;
    /// Supports FEC 3/4
    pub const FE_CAN_FEC_3_4: u32                   = 0x8;
    /// Supports FEC 4/5
    pub const FE_CAN_FEC_4_5: u32                   = 0x10;
    /// Supports FEC 5/6
    pub const FE_CAN_FEC_5_6: u32                   = 0x20;
    /// Supports FEC 6/7
    pub const FE_CAN_FEC_6_7: u32                   = 0x40;
    /// Supports FEC 7/8
    pub const FE_CAN_FEC_7_8: u32                   = 0x80;
    /// Supports FEC 8/9
    pub const FE_CAN_FEC_8_9: u32                   = 0x100;
    /// Can auto-detect FEC
    pub const FE_CAN_FEC_AUTO: u32                  = 0x200;
    /// Supports QPSK modulation
    pub const FE_CAN_QPSK: u32                      = 0x400;
    /// Supports 16-QAM modulation
    pub const FE_CAN_QAM_16: u32                    = 0x800;
    /// Supports 32-QAM modulation
    pub const FE_CAN_QAM_32: u32                    = 0x1000;
    /// Supports 64-QAM modulation
    pub const FE_CAN_QAM_64: u32                    = 0x2000;
    /// Supports 128-QAM modulation
    pub const FE_CAN_QAM_128: u32                   = 0x4000;
    /// Supports 256-QAM modulation
    pub const FE_CAN_QAM_256: u32                   = 0x8000;
    /// Can auto-detect QAM modulation
    pub const FE_CAN_QAM_AUTO: u32                  = 0x10000;
    /// Can auto-detect transmission mode
    pub const FE_CAN_TRANSMISSION_MODE_AUTO: u32    = 0x20000;
    /// Can auto-detect bandwidth
    pub const FE_CAN_BANDWIDTH_AUTO: u32            = 0x40000;
    /// Can auto-detect guard interval
    pub const FE_CAN_GUARD_INTERVAL_AUTO: u32       = 0x80000;
    /// Can auto-detect hierarchy
    pub const FE_CAN_HIERARCHY_AUTO: u32            = 0x100000;
    /// Supports 8-VSB modulation
    pub const FE_CAN_8VSB: u32                      = 0x200000;
    /// Supports 16-VSB modulation
    pub const FE_CAN_16VSB: u32                     = 0x400000;
    /// Unused
    pub const FE_HAS_EXTENDED_CAPS: u32             = 0x800000;
    /// Supports multistream filtering
    pub const FE_CAN_MULTISTREAM: u32               = 0x4000000;
    /// Supports "turbo FEC" modulation
    pub const FE_CAN_TURBO_FEC: u32                 = 0x8000000;
    /// Supports "2nd generation" modulation, e. g. DVB-S2, DVB-T2, DVB-C2
    pub const FE_CAN_2G_MODULATION: u32             = 0x10000000;
    /// Unused
    pub const FE_NEEDS_BENDING: u32                 = 0x20000000;
    /// Can recover from a cable unplug automatically
    pub const FE_CAN_RECOVER: u32                   = 0x40000000;
    /// Can stop spurious TS data output
    pub const FE_CAN_MUTE_TS: u32                   = 0x80000000;
}


/// DEPRECATED: Should be kept just due to backward compatibility
mod fe_type {
    pub const FE_QPSK: u32                      = 0;
    pub const FE_QAM: u32                       = 1;
    pub const FE_OFDM: u32                      = 2;
    pub const FE_ATSC: u32                      = 3;
}


/// Frontend properties and capabilities
/// The frequencies are specified in Hz for Terrestrial and Cable systems.
/// The frequencies are specified in kHz for Satellite systems.
#[repr(C)]
#[derive(Debug)]
pub struct FeInfo {
    /// Name of the frontend
    pub name: [std::os::raw::c_char; 128],
    /// DEPRECATED: frontend delivery system
    pub fe_type: u32,
    /// Minimal frequency supported by the frontend
    pub frequency_min: u32,
    /// Maximal frequency supported by the frontend
    pub frequency_max: u32,
    /// All frequencies are multiple of this value
    pub frequency_stepsize: u32,
    /// Frequency tolerance
    pub frequency_tolerance: u32,
    /// Minimal symbol rate, in bauds (for Cable/Satellite systems)
    pub symbol_rate_min: u32,
    /// Maximal symbol rate, in bauds (for Cable/Satellite systems)
    pub symbol_rate_max: u32,
    /// Maximal symbol rate tolerance, in ppm (for Cable/Satellite systems)
    pub symbol_rate_tolerance: u32,
    /// DEPRECATED
    pub notifier_delay: u32,
    /// Capabilities supported by the frontend
    pub caps: u32,
}


impl Default for FeInfo {
    #[inline]
    fn default() -> Self { unsafe { mem::zeroed::<Self>() } }
}


impl FeInfo {
    #[inline]
    pub fn as_mut_ptr(&mut self) -> *mut FeInfo { self as *mut _ }
}


/// DiSEqC master command
/// Check out the DiSEqC bus spec available on http://www.eutelsat.org/ for
/// the possible messages that can be used.
#[repr(C)]
#[derive(Debug)]
pub struct DiseqcMasterCmd {
    /// DiSEqC message to be sent. It contains a 3 bytes header with:
    /// framing + address + command, and an optional argument
    /// of up to 3 bytes of data.
    pub msg: [u8; 6],
    /// Length of the DiSEqC message. Valid values are 3 to 6.
    pub len: u8,
}


impl Default for DiseqcMasterCmd {
    #[inline]
    fn default() -> Self { unsafe { mem::zeroed::<Self>() } }
}


/// DiSEqC received data
#[repr(C)]
#[derive(Debug)]
pub struct DiseqcSlaveReply {
    /// DiSEqC message buffer to store a message received via DiSEqC.
    /// It contains one byte header with: framing and
    /// an optional argument of up to 3 bytes of data.
    pub msg: [u8; 4],
    /// Length of the DiSEqC message. Valid values are 0 to 4,
    /// where 0 means no message.
    pub len: u8,
    /// Return from ioctl after timeout ms with errorcode when
    /// no message was received.
    pub timeout: u32,
}


impl Default for DiseqcSlaveReply {
    #[inline]
    fn default() -> Self { unsafe { mem::zeroed::<Self>() } }
}


/// DC Voltage used to feed the LNBf
mod fe_sec_voltage {
    /// Output 13V to the LNB. Vertical linear. Right circular.
    pub const SEC_VOLTAGE_13: u32               = 0;
    /// Output 18V to the LNB. Horizontal linear. Left circular.
    pub const SEC_VOLTAGE_18: u32               = 1;
    /// Don't feed the LNB with a DC voltage
    pub const SEC_VOLTAGE_OFF: u32              = 2;
}


mod fe_sec_tone_mode {
    /// Sends a 22kHz tone burst to the antenna
    pub const SEC_TONE_ON: u32                  = 0;
    /// Don't send a 22kHz tone to the antenna (except if the FE_DISEQC_* ioctl are called)
    pub const SEC_TONE_OFF: u32                 = 1;
}


/// Type of mini burst to be sent
mod fe_sec_mini_cmd {
    /// Sends a mini-DiSEqC 22kHz '0' Tone Burst to select satellite-A
    pub const SEC_MINI_A: u32                   = 0;
    /// Sends a mini-DiSEqC 22kHz '1' Data Burst to select satellite-B
    pub const SEC_MINI_B: u32                   = 1;
}


/// Enumerates the possible frontend status
mod fe_status {
    /// The frontend doesn't have any kind of lock. That's the initial frontend status
    pub const FE_NONE: u32                      = 0x00;
    /// Has found something above the noise level
    pub const FE_HAS_SIGNAL: u32                = 0x01;
    /// Has found a signal
    pub const FE_HAS_CARRIER: u32               = 0x02;
    /// FEC inner coding (Viterbi, LDPC or other inner code) is stable.
    pub const FE_HAS_VITERBI: u32               = 0x04;
    /// Synchronization bytes was found
    pub const FE_HAS_SYNC: u32                  = 0x08;
    /// Digital TV were locked and everything is working
    pub const FE_HAS_LOCK: u32                  = 0x10;
    /// Fo lock within the last about 2 seconds
    pub const FE_TIMEDOUT: u32                  = 0x20;
    /// Frontend was reinitialized, application is recommended
    /// to reset DiSEqC, tone and parameters
    pub const FE_REINIT: u32                    = 0x40;
}


/// Spectral band inversion
mod fe_spectral_inversion {
    pub const INVERSION_OFF: u32                = 0;
    pub const INVERSION_ON: u32                 = 1;
    pub const INVERSION_AUTO: u32               = 2;
}


mod fe_code_rate {
    pub const FEC_NONE: u32                     = 0;
    pub const FEC_1_2: u32                      = 1;
    pub const FEC_2_3: u32                      = 2;
    pub const FEC_3_4: u32                      = 3;
    pub const FEC_4_5: u32                      = 4;
    pub const FEC_5_6: u32                      = 5;
    pub const FEC_6_7: u32                      = 6;
    pub const FEC_7_8: u32                      = 7;
    pub const FEC_8_9: u32                      = 8;
    pub const FEC_AUTO: u32                     = 9;
    pub const FEC_3_5: u32                      = 10;
    pub const FEC_9_10: u32                     = 11;
    pub const FEC_2_5: u32                      = 12;
    pub const FEC_1_4: u32                      = 13;
    pub const FEC_1_3: u32                      = 14;
}


/// Type of modulation/constellation
mod fe_modulation {
    pub const QPSK: u32                         = 0;
    pub const QAM_16: u32                       = 1;
    pub const QAM_32: u32                       = 2;
    pub const QAM_64: u32                       = 3;
    pub const QAM_128: u32                      = 4;
    pub const QAM_256: u32                      = 5;
    pub const QAM_AUTO: u32                     = 6;
    pub const VSB_8: u32                        = 7;
    pub const VSB_16: u32                       = 8;
    pub const PSK_8: u32                        = 9;
    pub const APSK_16: u32                      = 10;
    pub const APSK_32: u32                      = 11;
    pub const DQPSK: u32                        = 12;
    pub const QAM_4_NR: u32                     = 13;
    pub const APSK_64: u32                      = 14;
    pub const APSK_128: u32                     = 15;
    pub const APSK_256: u32                     = 16;
}


mod fe_transmit_mode {
    pub const TRANSMISSION_MODE_2K: u32         = 0;
    pub const TRANSMISSION_MODE_8K: u32         = 1;
    pub const TRANSMISSION_MODE_AUTO: u32       = 2;
    pub const TRANSMISSION_MODE_4K: u32         = 3;
    pub const TRANSMISSION_MODE_1K: u32         = 4;
    pub const TRANSMISSION_MODE_16K: u32        = 5;
    pub const TRANSMISSION_MODE_32K: u32        = 6;
    pub const TRANSMISSION_MODE_C1: u32         = 7;
    pub const TRANSMISSION_MODE_C3780: u32      = 8;
}


mod fe_guard_interval {
    pub const GUARD_INTERVAL_1_32: u32          = 0;
    pub const GUARD_INTERVAL_1_16: u32          = 1;
    pub const GUARD_INTERVAL_1_8: u32           = 2;
    pub const GUARD_INTERVAL_1_4: u32           = 3;
    pub const GUARD_INTERVAL_AUTO: u32          = 4;
    pub const GUARD_INTERVAL_1_128: u32         = 5;
    pub const GUARD_INTERVAL_19_128: u32        = 6;
    pub const GUARD_INTERVAL_19_256: u32        = 7;
    pub const GUARD_INTERVAL_PN420: u32         = 8;
    pub const GUARD_INTERVAL_PN595: u32         = 9;
    pub const GUARD_INTERVAL_PN945: u32         = 10;
}


mod fe_hierarchy {
    pub const HIERARCHY_NONE: u32               = 0;
    pub const HIERARCHY_1: u32                  = 1;
    pub const HIERARCHY_2: u32                  = 2;
    pub const HIERARCHY_4: u32                  = 3;
    pub const HIERARCHY_AUTO: u32               = 4;
}


mod fe_interleaving {
    pub const INTERLEAVING_NONE: u32            = 0;
    pub const INTERLEAVING_AUTO: u32            = 1;
    pub const INTERLEAVING_240: u32             = 2;
    pub const INTERLEAVING_720: u32             = 3;
}


mod fe_pilot {
    pub const PILOT_ON: u32                     = 0;
    pub const PILOT_OFF: u32                    = 1;
    pub const PILOT_AUTO: u32                   = 2;
}


mod fe_rolloff {
    pub const ROLLOFF_35: u32                   = 0;
    pub const ROLLOFF_20: u32                   = 1;
    pub const ROLLOFF_25: u32                   = 2;
    pub const ROLLOFF_AUTO: u32                 = 3;
    pub const ROLLOFF_15: u32                   = 4;
    pub const ROLLOFF_10: u32                   = 5;
    pub const ROLLOFF_5: u32                    = 6;
}


mod fe_delivery_system {
    use std::fmt;

    pub const SYS_UNDEFINED: u32                = 0;
    pub const SYS_DVBC_ANNEX_A: u32             = 1;
    pub const SYS_DVBC_ANNEX_B: u32             = 2;
    pub const SYS_DVBT: u32                     = 3;
    pub const SYS_DSS: u32                      = 4;
    pub const SYS_DVBS: u32                     = 5;
    pub const SYS_DVBS2: u32                    = 6;
    pub const SYS_DVBH: u32                     = 7;
    pub const SYS_ISDBT: u32                    = 8;
    pub const SYS_ISDBS: u32                    = 9;
    pub const SYS_ISDBC: u32                    = 10;
    pub const SYS_ATSC: u32                     = 11;
    pub const SYS_ATSCMH: u32                   = 12;
    pub const SYS_DTMB: u32                     = 13;
    pub const SYS_CMMB: u32                     = 14;
    pub const SYS_DAB: u32                      = 15;
    pub const SYS_DVBT2: u32                    = 16;
    pub const SYS_TURBO: u32                    = 17;
    pub const SYS_DVBC_ANNEX_C: u32             = 18;
    pub const SYS_DVBC2: u32                    = 19;

    pub struct DeliverySystemDisplay(pub u32);

    impl fmt::Display for DeliverySystemDisplay {
        fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
            let v = match self.0 {
                SYS_UNDEFINED => "none",
                SYS_DVBC_ANNEX_A => "dvb-c",
                SYS_DVBC_ANNEX_B => "dvb-c/b",
                SYS_DVBT => "dvb-t",
                SYS_DSS => "dss",
                SYS_DVBS => "dvb-s",
                SYS_DVBS2 => "dvb-s2",
                SYS_DVBH => "dvb-h",
                SYS_ISDBT => "isdb-t",
                SYS_ISDBS => "isdb-s",
                SYS_ISDBC => "isdb-c",
                SYS_ATSC => "atsc",
                SYS_ATSCMH => "atsc-m/h",
                SYS_DTMB => "dtmb",
                SYS_CMMB => "cmmb",
                SYS_DAB => "dab",
                SYS_DVBT2 => "dvb-t2",
                SYS_TURBO => "dvb-s/turbo",
                SYS_DVBC_ANNEX_C => "dvb-c/c",
                SYS_DVBC2 => "dvb-c2",
                _ => "unknown",
            };

            write!(f, "{}", v)
        }
    }
}


/// scale types for the quality parameters
mod fecap_scale_params {
    /// That QoS measure is not available. That could indicate
    /// a temporary or a permanent condition.
    pub const FE_SCALE_NOT_AVAILABLE: u8       = 0;
    /// The scale is measured in 0.001 dB steps, typically used on signal measures.
    pub const FE_SCALE_DECIBEL: u8             = 1;
    /// The scale is a relative percentual measure,
    /// ranging from 0 (0%) to 0xffff (100%).
    pub const FE_SCALE_RELATIVE: u8            = 2;
    /// The scale counts the occurrence of an event, like
    /// bit error, block error, lapsed time.
    pub const FE_SCALE_COUNTER: u8             = 3;
}


/// Used for reading a DTV status property
#[repr(C, packed)]
#[derive(Copy, Clone)]
pub struct DtvStats {
    pub scale: u8, // fecap_scale_params
    pub value: i64,
}


impl fmt::Debug for DtvStats {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        let mut s = f.debug_struct("DtvStats");

        const FIELD_SCALE: &str = "scale";
        const FIELD_VALUE: &str = "value";

        match self.scale {
            FE_SCALE_NOT_AVAILABLE => {
                s.field(FIELD_SCALE, &"FE_SCALE_NOT_AVAILABLE");
                s.field(FIELD_VALUE, &"not available");
            }
            FE_SCALE_DECIBEL => {
                s.field(FIELD_SCALE, &"FE_SCALE_DECIBEL");
                s.field(FIELD_VALUE, &{(self.value as f64) / 1000.0});
            }
            FE_SCALE_RELATIVE => {
                s.field(FIELD_SCALE, &"FE_SCALE_RELATIVE");
                s.field(FIELD_VALUE, &{self.value as u64});
            }
            FE_SCALE_COUNTER => {
                s.field(FIELD_SCALE, &"FE_SCALE_COUNTER");
                s.field(FIELD_VALUE, &{self.value as u64});
            }
            _ => {
                s.field(FIELD_SCALE, &{self.scale});
                s.field(FIELD_VALUE, &"invalid scale format");
            }
        };
        s.finish()
    }
}


pub const MAX_DTV_STATS: usize = 4;


/// Store Digital TV frontend statistics
#[repr(C, packed)]
#[derive(Copy, Clone)]
pub struct DtvFrontendStats {
    pub len: u8,
    pub stat: [DtvStats; MAX_DTV_STATS],
}


impl fmt::Debug for DtvFrontendStats {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        let len = ::std::cmp::min(self.len as usize, self.stat.len());
        f.debug_list().entries(self.stat[0 .. len].iter()).finish()
    }
}


#[repr(C)]
#[derive(Copy, Clone)]
pub struct DtvPropertyBuffer {
    pub data: [u8; 32],
    pub len: u32,
    __reserved_1: [u32; 3],
    __reserved_2: *mut std::ffi::c_void,
}


#[repr(C)]
pub union DtvPropertyData {
    pub data: u32,
    pub st: DtvFrontendStats,
    pub buffer: DtvPropertyBuffer,
    __align: [u8; 56],
}


/// DVBv5 property Commands
mod dtv_property_cmd {
    pub const DTV_UNDEFINED: u32                        = 0;
    pub const DTV_TUNE: u32                             = 1;
    pub const DTV_CLEAR: u32                            = 2;
    pub const DTV_FREQUENCY: u32                        = 3;
    pub const DTV_MODULATION: u32                       = 4;
    pub const DTV_BANDWIDTH_HZ: u32                     = 5;
    pub const DTV_INVERSION: u32                        = 6;
    pub const DTV_DISEQC_MASTER: u32                    = 7;
    pub const DTV_SYMBOL_RATE: u32                      = 8;
    pub const DTV_INNER_FEC: u32                        = 9;
    pub const DTV_VOLTAGE: u32                          = 10;
    pub const DTV_TONE: u32                             = 11;
    pub const DTV_PILOT: u32                            = 12;
    pub const DTV_ROLLOFF: u32                          = 13;
    pub const DTV_DISEQC_SLAVE_REPLY: u32               = 14;

    /* Basic enumeration set for querying unlimited capabilities */

    pub const DTV_FE_CAPABILITY_COUNT: u32              = 15;
    pub const DTV_FE_CAPABILITY: u32                    = 16;
    pub const DTV_DELIVERY_SYSTEM: u32                  = 17;

    /* ISDB-T and ISDB-Tsb */

    pub const DTV_ISDBT_PARTIAL_RECEPTION: u32          = 18;
    pub const DTV_ISDBT_SOUND_BROADCASTING: u32         = 19;

    pub const DTV_ISDBT_SB_SUBCHANNEL_ID: u32           = 20;
    pub const DTV_ISDBT_SB_SEGMENT_IDX: u32             = 21;
    pub const DTV_ISDBT_SB_SEGMENT_COUNT: u32           = 22;

    pub const DTV_ISDBT_LAYERA_FEC: u32                 = 23;
    pub const DTV_ISDBT_LAYERA_MODULATION: u32          = 24;
    pub const DTV_ISDBT_LAYERA_SEGMENT_COUNT: u32       = 25;
    pub const DTV_ISDBT_LAYERA_TIME_INTERLEAVING: u32   = 26;

    pub const DTV_ISDBT_LAYERB_FEC: u32                 = 27;
    pub const DTV_ISDBT_LAYERB_MODULATION: u32          = 28;
    pub const DTV_ISDBT_LAYERB_SEGMENT_COUNT: u32       = 29;
    pub const DTV_ISDBT_LAYERB_TIME_INTERLEAVING: u32   = 30;

    pub const DTV_ISDBT_LAYERC_FEC: u32                 = 31;
    pub const DTV_ISDBT_LAYERC_MODULATION: u32          = 32;
    pub const DTV_ISDBT_LAYERC_SEGMENT_COUNT: u32       = 33;
    pub const DTV_ISDBT_LAYERC_TIME_INTERLEAVING: u32   = 34;

    pub const DTV_API_VERSION: u32                      = 35;

    /* DVB-T/T2 */

    pub const DTV_CODE_RATE_HP: u32                     = 36;
    pub const DTV_CODE_RATE_LP: u32                     = 37;
    pub const DTV_GUARD_INTERVAL: u32                   = 38;
    pub const DTV_TRANSMISSION_MODE: u32                = 39;
    pub const DTV_HIERARCHY: u32                        = 40;

    pub const DTV_ISDBT_LAYER_ENABLED: u32              = 41;

    pub const DTV_STREAM_ID: u32                        = 42;
    pub const DTV_DVBT2_PLP_ID_LEGACY: u32              = 43;

    pub const DTV_ENUM_DELSYS: u32                      = 44;

    /* ATSC-MH */

    pub const DTV_ATSCMH_FIC_VER: u32                   = 45;
    pub const DTV_ATSCMH_PARADE_ID: u32                 = 46;
    pub const DTV_ATSCMH_NOG: u32                       = 47;
    pub const DTV_ATSCMH_TNOG: u32                      = 48;
    pub const DTV_ATSCMH_SGN: u32                       = 49;
    pub const DTV_ATSCMH_PRC: u32                       = 50;
    pub const DTV_ATSCMH_RS_FRAME_MODE: u32             = 51;
    pub const DTV_ATSCMH_RS_FRAME_ENSEMBLE: u32         = 52;
    pub const DTV_ATSCMH_RS_CODE_MODE_PRI: u32          = 53;
    pub const DTV_ATSCMH_RS_CODE_MODE_SEC: u32          = 54;
    pub const DTV_ATSCMH_SCCC_BLOCK_MODE: u32           = 55;
    pub const DTV_ATSCMH_SCCC_CODE_MODE_A: u32          = 56;
    pub const DTV_ATSCMH_SCCC_CODE_MODE_B: u32          = 57;
    pub const DTV_ATSCMH_SCCC_CODE_MODE_C: u32          = 58;
    pub const DTV_ATSCMH_SCCC_CODE_MODE_D: u32          = 59;

    pub const DTV_INTERLEAVING: u32                     = 60;
    pub const DTV_LNA: u32                              = 61;

    /* Quality parameters */

    pub const DTV_STAT_SIGNAL_STRENGTH: u32             = 62;
    pub const DTV_STAT_CNR: u32                         = 63;
    pub const DTV_STAT_PRE_ERROR_BIT_COUNT: u32         = 64;
    pub const DTV_STAT_PRE_TOTAL_BIT_COUNT: u32         = 65;
    pub const DTV_STAT_POST_ERROR_BIT_COUNT: u32        = 66;
    pub const DTV_STAT_POST_TOTAL_BIT_COUNT: u32        = 67;
    pub const DTV_STAT_ERROR_BLOCK_COUNT: u32           = 68;
    pub const DTV_STAT_TOTAL_BLOCK_COUNT: u32           = 69;

    /* Physical layer scrambling */

    pub const DTV_SCRAMBLING_SEQUENCE_INDEX: u32        = 70;
    pub const DTV_INPUT: u32                            = 71;
}


/// Store one of frontend command and its value
#[repr(C, packed)]
pub struct DtvProperty {
    pub cmd: u32,
    __reserved_1: [u32; 3],
    pub u: DtvPropertyData,
    pub result: i32,
}


impl fmt::Debug for DtvProperty {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        let mut s = f.debug_struct("DtvProperty");

        const FIELD_CMD: &str = "cmd";
        const FIELD_DATA: &str = "data";
        const FIELD_STATS: &str = "stats";

        match self.cmd {
            DTV_FREQUENCY => {
                s.field(FIELD_CMD, &"DTV_FREQUENCY");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }
            DTV_MODULATION => {
                s.field(FIELD_CMD, &"DTV_MODULATION");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }
            DTV_BANDWIDTH_HZ => {
                s.field(FIELD_CMD, &"DTV_BANDWIDTH_HZ");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }
            DTV_INVERSION => {
                s.field(FIELD_CMD, &"DTV_INVERSION");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }
            DTV_SYMBOL_RATE  => {
                s.field(FIELD_CMD, &"DTV_SYMBOL_RATE");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }
            DTV_INNER_FEC => {
                s.field(FIELD_CMD, &"DTV_INNER_FEC");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }
            DTV_PILOT => {
                s.field(FIELD_CMD, &"DTV_PILOT");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }
            DTV_ROLLOFF => {
                s.field(FIELD_CMD, &"DTV_ROLLOFF");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }
            DTV_DELIVERY_SYSTEM => {
                s.field(FIELD_CMD, &"DTV_DELIVERY_SYSTEM");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }
            DTV_API_VERSION => {
                s.field(FIELD_CMD, &"DTV_API_VERSION");
                s.field(FIELD_DATA, unsafe { &self.u.data });
            }

            /* Quality parameters */

            DTV_STAT_SIGNAL_STRENGTH => {
                s.field(FIELD_CMD, &"DTV_STAT_SIGNAL_STRENGTH");
                s.field(FIELD_STATS, unsafe { &self.u.st });
            }
            DTV_STAT_CNR => {
                s.field(FIELD_CMD, &"DTV_STAT_CNR");
                s.field(FIELD_STATS, unsafe { &self.u.st });
            }

            DTV_STAT_PRE_ERROR_BIT_COUNT => {
                s.field(FIELD_CMD, &"DTV_STAT_PRE_ERROR_BIT_COUNT");
                s.field(FIELD_STATS, unsafe { &self.u.st });
            }
            DTV_STAT_PRE_TOTAL_BIT_COUNT => {
                s.field(FIELD_CMD, &"DTV_STAT_PRE_TOTAL_BIT_COUNT");
                s.field(FIELD_STATS, unsafe { &self.u.st });
            }
            DTV_STAT_POST_ERROR_BIT_COUNT => {
                s.field(FIELD_CMD, &"DTV_STAT_POST_ERROR_BIT_COUNT");
                s.field(FIELD_STATS, unsafe { &self.u.st });
            }
            DTV_STAT_POST_TOTAL_BIT_COUNT => {
                s.field(FIELD_CMD, &"DTV_STAT_POST_TOTAL_BIT_COUNT");
                s.field(FIELD_STATS, unsafe { &self.u.st });
            }
            DTV_STAT_ERROR_BLOCK_COUNT => {
                s.field(FIELD_CMD, &"DTV_STAT_ERROR_BLOCK_COUNT");
                s.field(FIELD_STATS, unsafe { &self.u.st });
            }
            DTV_STAT_TOTAL_BLOCK_COUNT => {
                s.field(FIELD_CMD, &"DTV_STAT_TOTAL_BLOCK_COUNT");
                s.field(FIELD_STATS, unsafe { &self.u.st });
            }

            // TODO: more values
            _ => {}
        }

        s.field("result", &{ self.result });
        s.finish()
    }
}


impl DtvProperty {
    #[inline]
    pub fn new(cmd: u32, data: u32) -> Self {
        Self {
            cmd,
            __reserved_1: [0, 0, 0],
            u: DtvPropertyData { data },
            result: 0,
        }
    }
}


pub const DTV_MAX_COMMAND: u32                          = DTV_INPUT;


/// num of properties cannot exceed DTV_IOCTL_MAX_MSGS per ioctl
pub const DTV_IOCTL_MAX_MSGS: usize                     = 64;


#[repr(C)]
#[derive(Debug)]
pub struct FeParameters {
    /// (absolute) frequency in Hz for DVB-C/DVB-T/ATSC
    /// intermediate frequency in kHz for DVB-S
    pub frequency: u32,
    pub inversion: u32,
    /// unimplemented frontend parameters data
    __reserved_1: [u8; 28],
}


pub const FE_MAX_EVENT: usize = 8;


#[repr(C)]
#[derive(Debug)]
pub struct FeEvent {
    pub status: u32,
    pub parameters: FeParameters,
}


impl Default for FeEvent {
    #[inline]
    fn default() -> Self { unsafe { mem::zeroed::<Self>() } }
}


impl FeEvent {
    #[inline]
    pub fn as_mut_ptr(&mut self) -> *mut FeEvent { self as *mut _ }
}