1
|
# Copyright 2010 Dan Smith <dsmith@danplanet.com>
|
2
|
# Copyright 2017 Nicolas Pike <nick@zbm2.com>
|
3
|
#
|
4
|
# This program is free software: you can redistribute it and/or modify
|
5
|
# it under the terms of the GNU General Public License as published by
|
6
|
# the Free Software Foundation, either version 3 of the License, or
|
7
|
# (at your option) any later version.
|
8
|
#
|
9
|
# This program is distributed in the hope that it will be useful,
|
10
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
11
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
12
|
# GNU General Public License for more details.
|
13
|
#
|
14
|
# You should have received a copy of the GNU General Public License
|
15
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
16
|
|
17
|
import logging
|
18
|
|
19
|
from chirp.drivers import yaesu_clone
|
20
|
from chirp import chirp_common, directory, bitwise
|
21
|
from chirp import errors
|
22
|
from chirp import memmap
|
23
|
from chirp.settings import RadioSettingGroup, RadioSetting, RadioSettings, \
|
24
|
RadioSettingValueString, RadioSettingValueList, \
|
25
|
InvalidValueError
|
26
|
from chirp import util
|
27
|
|
28
|
import string
|
29
|
LOG = logging.getLogger(__name__)
|
30
|
|
31
|
# Testing
|
32
|
|
33
|
# 37 PAG.ABK Turn the pager answer back Function ON/OFF
|
34
|
# 38 PAG.CDR Specify a personal code (receive)
|
35
|
# 39 PAG.CDT Specify a personal code (transmit)
|
36
|
# 47 RX.MOD Select the receive mode. Auto FM AM
|
37
|
|
38
|
MEM_SETTINGS_FORMAT = """
|
39
|
|
40
|
// FT-70DE New Model #5329
|
41
|
//
|
42
|
// Communications Mode ? AMS,FM DN,DW TX vs RX?
|
43
|
// Mode not currently correctly stored in memories ? - ALL show as FM in memories
|
44
|
// SKIP test/where stored
|
45
|
// Check storage of steps
|
46
|
// Pager settings ?
|
47
|
// Triple check/ understand _memsize and _block_lengths
|
48
|
// Bank name label name size display 6 store 16? padded with 0xFF same for MYCALL and message
|
49
|
// CHIRP mode DIG not supported - is there a CHIRP Fusion mode? Auto?
|
50
|
// Check character set
|
51
|
// Supported Modes ?
|
52
|
// Supported Bands ?
|
53
|
// rf.has_dtcs_polarity = False - think radio supports DTCS polarity
|
54
|
// rf.memory_bounds = (1, 900) - should this be 0? as zero displays as blank
|
55
|
// RPT offsets (stored per band) not included.
|
56
|
// 59 Display radio firmware version info and Radio ID?
|
57
|
// Front Panel settings power etc?
|
58
|
// Banks and VFO?
|
59
|
|
60
|
// Features Required
|
61
|
// Default AMS and Memory name (in mem extras) to enabled.
|
62
|
|
63
|
// Bugs
|
64
|
// MYCALL and Opening Message errors if not 10 characters
|
65
|
// Values greater than one sometimes stored as whole bytes, these need to be refactored into bit fields
|
66
|
// to prevent accidental overwriting of adjacent values
|
67
|
// Bank Name length not checked on gui input - but first 6 characters are saved correctly.
|
68
|
// Extended characters entered as bank names on radio are corrupted in Chirp
|
69
|
|
70
|
// Missing
|
71
|
// 50 SCV.WTH Set the VFO scan frequency range. BAND / ALL - NOT FOUND
|
72
|
// 49 SCM.WTH Set the memory scan frequency range. ALL / BAND - NOT FOUND
|
73
|
|
74
|
// Radio Questions
|
75
|
// Temp unit C/F not saved by radio, always goes back to C ?
|
76
|
// 44 RF SQL Adjusts the RF Squelch threshold level. OFF / S1 - S9? Default is OFF - Based on RF strength - for AM? How
|
77
|
// is this different from F, Monitor, Dial Squelch?
|
78
|
// Password setting on radio allows letters (DTMF), but letters cannot be entered at the radio's password prompt?
|
79
|
// 49 SCM.WTH Set the memory scan frequency range. ALL / BAND Defaults to ALL Not Band as stated in the manual.
|
80
|
|
81
|
#seekto 0x049a;
|
82
|
struct {
|
83
|
u8 unknown0:4,
|
84
|
squelch:4; // Squelch F, Monitor, Dial Adjust the Squelch level
|
85
|
} squelch_settings;
|
86
|
|
87
|
#seekto 0x04ba;
|
88
|
struct {
|
89
|
u8 unknown:3,
|
90
|
scan_resume:5; // 52 SCN.RSM Configure the scan stop mode settings. 2.0 S - 5.0 S - 10.0 S / BUSY / HOLD
|
91
|
u8 unknown1:3,
|
92
|
dw_resume_interval:5; // 22 DW RSM Configure the scan stop mode settings for Dual Receive. 2.0S-10.0S/BUSY/HOLD
|
93
|
u8 unknown2;
|
94
|
u8 unknown3:3,
|
95
|
apo:5; // 02 APO Set the length of time until the transceiver turns off automatically.
|
96
|
u8 unknown4:6,
|
97
|
gm_ring:2; // 24 GM RNG Select the beep option while receiving digital GM info. OFF/IN RNG/ALWAYS
|
98
|
u8 temp_cf; // Placeholder as not found
|
99
|
u8 unknown5;
|
100
|
} first_settings;
|
101
|
|
102
|
#seekto 0x04ed;
|
103
|
struct {
|
104
|
u8 unknown1:1,
|
105
|
unknown2:1,
|
106
|
unknown3:1,
|
107
|
unknown4:1,
|
108
|
unknown5:1,
|
109
|
unknown6:1,
|
110
|
unknown7:1,
|
111
|
unknown8:1;
|
112
|
} test_bit_field;
|
113
|
|
114
|
#seekto 0x04c0;
|
115
|
struct {
|
116
|
u8 unknown1:5,
|
117
|
beep_level:3; // 05 BEP.LVL Beep volume setting LEVEL1 - LEVEL4 - LEVEL7
|
118
|
u8 unknown2:6,
|
119
|
beep_select:2; // 04 BEEP Sets the beep sound function OFF / KEY+SC / KEY
|
120
|
} beep_settings;
|
121
|
|
122
|
#seekto 0x04ce;
|
123
|
struct {
|
124
|
u8 lcd_dimmer; // 14 DIMMER LCD Dimmer
|
125
|
u8 dtmf_delay; // 18 DT DLY DTMF delay
|
126
|
u8 unknown0[3];
|
127
|
u8 unknown1:4,
|
128
|
unknown1_2:4;
|
129
|
u8 lamp; // 28 LAMP Set the duration time of the backlight and keys to be lit
|
130
|
u8 lock; // 30 LOCK Configure the lock mode setting. KEY/DIAL/K+D/PTT/K+P/D+P/ALL
|
131
|
u8 unknown2_1;
|
132
|
u8 mic_gain; // 31 MCGAIN Adjust the microphone gain level
|
133
|
u8 unknown2_3;
|
134
|
u8 dw_interval; // 21 DW INT Set the priority memory ch mon int during Dual RX 0.1S-5.0S-10.0S
|
135
|
u8 ptt_delay; // 42 PTT.DLY Set the PTT delay time. OFF / 20 ms / 50 ms / 100 ms / 200 ms
|
136
|
u8 rx_save; // 48 RX.SAVE Set the battery save time. OFF / 0.2 s - 60.0 s
|
137
|
u8 scan_restart; // 53 SCN.STR Set the scanning restart time. 0.1 s - 2.0 s - 10.0 s
|
138
|
u8 unknown2_5;
|
139
|
u8 unknown2_6;
|
140
|
u8 unknown4[5];
|
141
|
u8 tot; // 56 TOT Set the transmission timeout timer
|
142
|
u8 unknown5[3]; // 26
|
143
|
u8 vfo_mode:1, // 60 VFO.MOD Set freq setting range in the VFO mode by DIAL knob. ALL / BAND
|
144
|
unknown7:1,
|
145
|
scan_lamp:1, // 51 SCN.LMP Set the scan lamp ON or OFF when scanning stops On/Off
|
146
|
unknown8:1,
|
147
|
ars:1, // 45 RPT.ARS Turn the ARS function on/off.
|
148
|
dtmf_speed:1, // 20 DT SPD Set DTMF speed
|
149
|
unknown8_1:1,
|
150
|
dtmf_mode:1; // DTMF Mode set from front panel
|
151
|
u8 busy_led:1, // Not Supported ?
|
152
|
unknown8_2:1,
|
153
|
unknown8_3:1,
|
154
|
bclo:1, // 03 BCLO Turns the busy channel lockout function on/off.
|
155
|
beep_edge:1, // 06 BEP.Edg Sets the beep sound ON or OFF when a band edge is encountered.
|
156
|
unknown8_6:1,
|
157
|
unknown8_7:1,
|
158
|
unknown8_8:1; // 28
|
159
|
u8 unknown9_1:1,
|
160
|
unknown9_2:1,
|
161
|
unknown9_3:1,
|
162
|
unknown9_4:1,
|
163
|
unknown9_5:1,
|
164
|
password:1, // Placeholder location
|
165
|
home_rev:1, // 26 HOME/REV Select the function of the [HOME/REV] key.
|
166
|
moni:1; // 32 Mon/T-Call Select the function of the [MONI/T-CALL] switch.
|
167
|
u8 gm_interval:4, // 30 // 25 GM INT Set tx interval of digital GM information. OFF / NORMAL / LONG
|
168
|
unknown10:4;
|
169
|
u8 unknown11;
|
170
|
u8 unknown12:1,
|
171
|
unknown12_2:1,
|
172
|
unknown12_3:1,
|
173
|
unknown12_4:1,
|
174
|
home_vfo:1, // 27 HOME->VFO Turn transfer VFO to the Home channel ON or OFF.
|
175
|
unknown12_6:1,
|
176
|
unknown12_7:1,
|
177
|
dw_rt:1; // 32 // 23 DW RVT Turn "Priority Channel Revert" feature ON or OFF during Dual Rx.
|
178
|
u8 unknown33;
|
179
|
u8 unknown34;
|
180
|
u8 unknown35;
|
181
|
u8 unknown36;
|
182
|
u8 unknown37;
|
183
|
u8 unknown38;
|
184
|
u8 unknown39;
|
185
|
u8 unknown40;
|
186
|
u8 unknown41;
|
187
|
u8 unknown42;
|
188
|
u8 unknown43;
|
189
|
u8 unknown44;
|
190
|
u8 unknown45;
|
191
|
u8 prog_key1; // P1 Set Mode Items to the Programmable Key
|
192
|
u8 prog_key2; // P2 Set Mode Items to the Programmable Key
|
193
|
u8 unknown48;
|
194
|
u8 unknown49;
|
195
|
u8 unknown50;
|
196
|
} scan_settings;
|
197
|
|
198
|
#seekto 0x064b;
|
199
|
struct {
|
200
|
u8 unknown1:1,
|
201
|
unknown2:1,
|
202
|
unknown3:1,
|
203
|
unknown4:1,
|
204
|
vfo_scan_width:1, // Placeholder as not found - 50 SCV.WTH Set the VFO scan frequency range. BAND / ALL
|
205
|
memory_scan_width:1, // Placeholder as not found - 49 SCM.WTH Set the memory scan frequency range. ALL / BAND
|
206
|
unknown7:1,
|
207
|
unknown8:1;
|
208
|
} scan_settings_1;
|
209
|
|
210
|
#seekto 0x06B6;
|
211
|
struct {
|
212
|
u8 unknown1:3,
|
213
|
volume:5; // # VOL and Dial Adjust the volume level
|
214
|
} scan_settings_2;
|
215
|
|
216
|
#seekto 0x0690; // Memory or VFO Settings Map?
|
217
|
struct {
|
218
|
u8 unknown[48]; // Array cannot be 64 elements!
|
219
|
u8 unknown1[16]; // Exception: Not implemented for chirp.bitwise.structDataElement
|
220
|
} vfo_info_1;
|
221
|
|
222
|
#seekto 0x0710; // Backup Memory or VFO Settings Map?
|
223
|
struct {
|
224
|
u8 unknown[48];
|
225
|
u8 unknown1[16];
|
226
|
} vfo_backup_info_1;
|
227
|
|
228
|
#seekto 0x047e;
|
229
|
struct {
|
230
|
u8 unknown1;
|
231
|
u8 flag;
|
232
|
u16 unknown2;
|
233
|
struct {
|
234
|
char padded_string[6]; // 36 OPN.MSG Select MSG then key vm to edit it
|
235
|
} message;
|
236
|
} opening_message; // 36 OPN.MSG Select the Opening Message when transceiver is ON. OFF/MSG/DC
|
237
|
|
238
|
#seekto 0x094a; // DTMF Memories
|
239
|
struct {
|
240
|
u8 memory[16];
|
241
|
} dtmf[10];
|
242
|
|
243
|
#seekto 0x154a;
|
244
|
struct {
|
245
|
u16 channel[100];
|
246
|
} bank_members[24];
|
247
|
|
248
|
#seekto 0x54a;
|
249
|
struct {
|
250
|
u16 in_use;
|
251
|
} bank_used[24];
|
252
|
|
253
|
#seekto 0x0EFE;
|
254
|
struct {
|
255
|
u8 unknown[2];
|
256
|
u8 name[6];
|
257
|
u8 unknown1[10];
|
258
|
} bank_info[24];
|
259
|
|
260
|
#seekto 0xCF30;
|
261
|
struct {
|
262
|
u8 unknown0;
|
263
|
u8 unknown1;
|
264
|
u8 unknown2;
|
265
|
u8 unknown3;
|
266
|
u8 unknown4;
|
267
|
u8 unknown5;
|
268
|
u8 unknown6;
|
269
|
u8 digital_popup; // 15 DIG.POP Call sign display pop up time
|
270
|
} digital_settings_more;
|
271
|
|
272
|
#seekto 0xCF7C;
|
273
|
struct {
|
274
|
u8 unknown0:6,
|
275
|
ams_tx_mode:2; // AMS TX Mode Short Press AMS button AMS TX Mode
|
276
|
u8 unknown1;
|
277
|
u8 unknown2:7,
|
278
|
standby_beep:1; // 07 BEP.STB Standby Beep in the digital C4FM mode. On/Off
|
279
|
u8 unknown3;
|
280
|
u8 unknown4:6,
|
281
|
gm_ring:2; // 24 GM RNG Select beep option while rx digital GM info. OFF/IN RNG/ALWAYS
|
282
|
u8 unknown5;
|
283
|
u8 rx_dg_id; // RX DG-ID Long Press Mode Key, Mode Key to select, Dial
|
284
|
u8 tx_dg_id; // TX DG-ID Long Press Mode Key, Dial
|
285
|
u8 unknown6:7,
|
286
|
vw_mode:1; // 16 DIG VW Turn the VW mode selection ON or OFF
|
287
|
u8 unknown7;
|
288
|
} digital_settings;
|
289
|
|
290
|
// ^^^ All above referenced U8's have been refactored to minimum number of bits.
|
291
|
|
292
|
"""
|
293
|
|
294
|
MEM_FORMAT = """
|
295
|
#seekto 0x2D4A;
|
296
|
struct { // 32 Bytes per memory entry
|
297
|
u8 display_tag:1, // 0 Display Freq, 1 Display Name
|
298
|
unknown0:1, // Mode if AMS not selected????????
|
299
|
deviation:1, // 0 Full deviation (Wide), 1 Half deviation (Narrow)
|
300
|
clock_shift:1, // 0 None, 1 CPU clock shifted
|
301
|
unknown1:4; // 1
|
302
|
u8 mode:2, // FM,AM,WFM only? - check
|
303
|
duplex:2, // Works
|
304
|
tune_step:4; // Works - check all steps? 7 = Auto // 1
|
305
|
bbcd freq[3]; // Works // 3
|
306
|
u8 power:2, // Works
|
307
|
unknown2:1, // 0 FM, 1 Digital - If AMS off
|
308
|
ams:1, // 0 AMS off, 1 AMS on ?
|
309
|
tone_mode:4; // Works // 1
|
310
|
u8 charsetbits[2]; // 2
|
311
|
char label[6]; // Works - Can only input 6 on screen // 6
|
312
|
char unknown7[10]; // Rest of label ??? // 10
|
313
|
bbcd offset[3]; // Works // 3
|
314
|
u8 unknown5:2,
|
315
|
tone:6; // Works // 1
|
316
|
u8 unknown6:1,
|
317
|
dcs:7; // Works // 1
|
318
|
u8 unknown9;
|
319
|
u8 ams_on_dn_vw_fm:2, // AMS DN, AMS VW, AMS FM
|
320
|
unknown8_3:1,
|
321
|
unknown8_4:1,
|
322
|
smeter:4;
|
323
|
u8 unknown10:2,
|
324
|
att:1,
|
325
|
auto_step:1,
|
326
|
auto_mode:1,
|
327
|
unknown11:2,
|
328
|
bell:1;
|
329
|
} memory[%d]; // DN, VW, FM, AM
|
330
|
// AMS DN, AMS VW, AMS FM
|
331
|
|
332
|
#seekto 0x280A;
|
333
|
struct {
|
334
|
u8 nosubvfo:1,
|
335
|
unknown:3,
|
336
|
pskip:1, // PSkip (Select?)
|
337
|
skip:1, // Skip memory during scan
|
338
|
used:1, // Memory used
|
339
|
valid:1; // Always 1?
|
340
|
} flag[%d];
|
341
|
"""
|
342
|
|
343
|
MEM_CALLSIGN_FORMAT = """
|
344
|
#seekto 0x0ced0;
|
345
|
struct {
|
346
|
char callsign[10]; // 63 MYCALL Set the call sign. (up to 10 characters)
|
347
|
u16 charset; // character set ID
|
348
|
} my_call;
|
349
|
"""
|
350
|
|
351
|
MEM_CHECKSUM_FORMAT = """
|
352
|
#seekto 0xFECA;
|
353
|
u8 checksum;
|
354
|
"""
|
355
|
|
356
|
TMODES = ["", "Tone", "TSQL", "DTCS"]
|
357
|
DUPLEX = ["", "-", "+", "split"]
|
358
|
|
359
|
MODES = ["FM", "AM", "NFM"]
|
360
|
|
361
|
STEPS = [0, 5, 6.25, 10, 12.5, 15, 20, 25, 50, 100] # 0 = auto
|
362
|
RFSQUELCH = ["OFF", "S1", "S2", "S3", "S4", "S5", "S6", "S7", "S8"]
|
363
|
|
364
|
SKIPS = ["", "S", "P"]
|
365
|
FT70_DTMF_CHARS = list("0123456789ABCDEF-")
|
366
|
|
367
|
CHARSET = ["%i" % int(x) for x in range(0, 10)] + \
|
368
|
[chr(x) for x in range(ord("A"), ord("Z") + 1)] + \
|
369
|
[" ", ] + \
|
370
|
[chr(x) for x in range(ord("a"), ord("z") + 1)] + \
|
371
|
list(".,:;*#_-/&()@!?^ ") + list("\x00" * 100)
|
372
|
|
373
|
POWER_LEVELS = [chirp_common.PowerLevel("Hi", watts=5.00),
|
374
|
chirp_common.PowerLevel("Mid", watts=2.00),
|
375
|
chirp_common.PowerLevel("Low", watts=.50)]
|
376
|
|
377
|
|
378
|
class FT70Bank(chirp_common.NamedBank):
|
379
|
"""A FT70 bank"""
|
380
|
|
381
|
def get_name(self):
|
382
|
_bank = self._model._radio._memobj.bank_info[self.index]
|
383
|
name = ""
|
384
|
for i in _bank.name:
|
385
|
if i == 0xff:
|
386
|
break
|
387
|
name += chr(i & 0xFF)
|
388
|
return name.rstrip()
|
389
|
|
390
|
def set_name(self, name):
|
391
|
_bank = self._model._radio._memobj.bank_info[self.index]
|
392
|
_bank.name = [ord(x) for x in name.ljust(6, chr(0xFF))[:6]]
|
393
|
|
394
|
|
395
|
class FT70BankModel(chirp_common.BankModel):
|
396
|
"""A FT70 bank model"""
|
397
|
|
398
|
def __init__(self, radio, name='Banks'):
|
399
|
super(FT70BankModel, self).__init__(radio, name)
|
400
|
|
401
|
_banks = self._radio._memobj.bank_info
|
402
|
self._bank_mappings = []
|
403
|
for index, _bank in enumerate(_banks):
|
404
|
bank = FT70Bank(self, "%i" % index, "BANK-%i" % index)
|
405
|
bank.index = index
|
406
|
self._bank_mappings.append(bank)
|
407
|
|
408
|
def get_num_mappings(self):
|
409
|
return len(self._bank_mappings)
|
410
|
|
411
|
def get_mappings(self):
|
412
|
return self._bank_mappings
|
413
|
|
414
|
def _channel_numbers_in_bank(self, bank):
|
415
|
_bank_used = self._radio._memobj.bank_used[bank.index]
|
416
|
if _bank_used.in_use == 0xFFFF:
|
417
|
return set()
|
418
|
|
419
|
_members = self._radio._memobj.bank_members[bank.index]
|
420
|
return set([int(ch) + 1 for ch in _members.channel if ch != 0xFFFF])
|
421
|
|
422
|
def _update_bank_with_channel_numbers(self, bank, channels_in_bank):
|
423
|
_members = self._radio._memobj.bank_members[bank.index]
|
424
|
if len(channels_in_bank) > len(_members.channel):
|
425
|
raise Exception("Too many entries in bank %d" % bank.index)
|
426
|
|
427
|
empty = 0
|
428
|
for index, channel_number in enumerate(sorted(channels_in_bank)):
|
429
|
_members.channel[index] = channel_number - 1
|
430
|
empty = index + 1
|
431
|
for index in range(empty, len(_members.channel)):
|
432
|
_members.channel[index] = 0xFFFF
|
433
|
|
434
|
def add_memory_to_mapping(self, memory, bank):
|
435
|
channels_in_bank = self._channel_numbers_in_bank(bank)
|
436
|
channels_in_bank.add(memory.number)
|
437
|
self._update_bank_with_channel_numbers(bank, channels_in_bank)
|
438
|
|
439
|
_bank_used = self._radio._memobj.bank_used[bank.index]
|
440
|
_bank_used.in_use = 0x06
|
441
|
|
442
|
def remove_memory_from_mapping(self, memory, bank):
|
443
|
channels_in_bank = self._channel_numbers_in_bank(bank)
|
444
|
try:
|
445
|
channels_in_bank.remove(memory.number)
|
446
|
except KeyError:
|
447
|
raise Exception("Memory %i is not in bank %s. Cannot remove" %
|
448
|
(memory.number, bank))
|
449
|
self._update_bank_with_channel_numbers(bank, channels_in_bank)
|
450
|
|
451
|
if not channels_in_bank:
|
452
|
_bank_used = self._radio._memobj.bank_used[bank.index]
|
453
|
_bank_used.in_use = 0xFFFF
|
454
|
|
455
|
def get_mapping_memories(self, bank):
|
456
|
memories = []
|
457
|
for channel in self._channel_numbers_in_bank(bank):
|
458
|
memories.append(self._radio.get_memory(channel))
|
459
|
|
460
|
return memories
|
461
|
|
462
|
def get_memory_mappings(self, memory):
|
463
|
banks = []
|
464
|
for bank in self.get_mappings():
|
465
|
if memory.number in self._channel_numbers_in_bank(bank):
|
466
|
banks.append(bank)
|
467
|
|
468
|
return banks
|
469
|
|
470
|
|
471
|
@directory.register
|
472
|
class FT70Radio(yaesu_clone.YaesuCloneModeRadio):
|
473
|
"""Yaesu FT-70DE"""
|
474
|
BAUD_RATE = 38400
|
475
|
VENDOR = "Yaesu"
|
476
|
MODEL = "FT-70D"
|
477
|
FORMATS = [directory.register_format('FT-70D ADMS-10', '*.ft70d')]
|
478
|
|
479
|
_model = b"AH51G"
|
480
|
_adms_ext = '.ft70d'
|
481
|
|
482
|
_memsize = 65227 # 65227 read from dump
|
483
|
_block_lengths = [10, 65217]
|
484
|
_block_size = 32
|
485
|
_mem_params = (900, # size of memories array
|
486
|
900, # size of flags array
|
487
|
)
|
488
|
|
489
|
_has_vibrate = False
|
490
|
_has_af_dual = True
|
491
|
|
492
|
_BEEP_SELECT = ("Off", "Key+Scan", "Key")
|
493
|
_OPENING_MESSAGE = ("Off", "DC", "Message")
|
494
|
_MIC_GAIN = ("Level 1", "Level 2", "Level 3", "Level 4", "Level 5", "Level 6", "Level 7", "Level 8", "Level 9")
|
495
|
_AMS_TX_MODE = ("TX Auto", "TX DIGITAL", "TX FM")
|
496
|
_VW_MODE = ("On", "Off")
|
497
|
_DIG_POP_UP = ("Off", "2sec", "4sec", "6sec", "8sec", "10sec", "20sec", "30sec", "60sec", "Continuous")
|
498
|
_STANDBY_BEEP = ("On", "Off")
|
499
|
_SCAN_RESUME = ["%.1fs" % (0.5 * x) for x in range(4, 21)] + \
|
500
|
["Busy", "Hold"]
|
501
|
_SCAN_RESTART = ["%.1fs" % (0.1 * x) for x in range(1, 10)] + \
|
502
|
["%.1fs" % (0.5 * x) for x in range(2, 21)]
|
503
|
_LAMP_KEY = ["Key %d sec" % x
|
504
|
for x in range(2, 11)] + ["Continuous", "OFF"]
|
505
|
_LCD_DIMMER = ["Level %d" % x for x in range(1, 7)]
|
506
|
_TOT_TIME = ["Off"] + ["%.1f min" % (0.5 * x) for x in range(1, 21)]
|
507
|
_OFF_ON = ("Off", "On")
|
508
|
_ON_OFF = ("On", "Off")
|
509
|
_DTMF_MODE = ("Manual", "Auto")
|
510
|
_DTMF_SPEED = ("50ms", "100ms")
|
511
|
_DTMF_DELAY = ("50ms", "250ms", "450ms", "750ms", "1000ms")
|
512
|
_TEMP_CF = ("Centigrade", "Fahrenheit")
|
513
|
_APO_SELECT = ("Off", "0.5H", "1.0H", "1.5H", "2.0H", "2.5H", "3.0H", "3.5H", "4.0H", "4.5H", "5.0H",
|
514
|
"5.5H", "6.0H", "6.5H", "7.0H", "7.5H", "8.0H", "8.5H", "9.0H", "9.5H", "10.0H", "10.5H",
|
515
|
"11.0H", "11.5H", "12.0H")
|
516
|
_MONI_TCALL = ("Monitor", "Tone-CALL")
|
517
|
_HOME_REV = ("Home", "Reverse")
|
518
|
_LOCK = ("KEY", "DIAL", "Key+Dial", "PTT", "Key+PTT", "Dial+PTT", "ALL")
|
519
|
_PTT_DELAY = ("Off", "20 ms", "50 ms", "100 ms", "200 ms")
|
520
|
_BEEP_LEVEL = ("Level 1", "Level 2", "Level 3", "Level 4", "Level 5", "Level 6", "Level 7")
|
521
|
_SET_MODE = ("Level 1", "Level 2", "Level 3", "Level 4", "Level 5", "Level 6", "Level 7")
|
522
|
_RX_SAVE = ("OFF", "0.2s", ".3s", ".4s", ".5s", ".6s", ".7s", ".8s", ".9s", "1.0s", "1.5s",
|
523
|
"2.0s", "2.5s", "3.0s", "3.5s", "4.0s", "4.5s", "5.0s", "5.5s", "6.0s", "6.5s", "7.0s",
|
524
|
"7.5s", "8.0s", "8.5s", "9.0s", "10.0s", "15s", "20s", "25s", "30s", "35s", "40s", "45s", "50s", "55s",
|
525
|
"60s")
|
526
|
_VFO_MODE = ("ALL", "BAND")
|
527
|
_VFO_SCAN_MODE = ("BAND", "ALL")
|
528
|
_MEMORY_SCAN_MODE = ("BAND", "ALL")
|
529
|
|
530
|
_VOLUME = ["Level %d" % x for x in range(0, 32)]
|
531
|
_SQUELCH = ["Level %d" % x for x in range(0, 16)]
|
532
|
|
533
|
_DG_ID = ["%d" % x for x in range(0, 100)]
|
534
|
_GM_RING = ("OFF", "IN RING", "AlWAYS")
|
535
|
_GM_INTERVAL = ("LONG", "NORMAL", "OFF")
|
536
|
|
537
|
_MYCALL_CHR_SET = list(string.ascii_uppercase) + list(string.digits) + ['-', '/']
|
538
|
|
539
|
@classmethod
|
540
|
def match_model(cls, filedata, filename):
|
541
|
if filename.endswith(cls._adms_ext):
|
542
|
return True
|
543
|
else:
|
544
|
return super().match_model(filedata, filename)
|
545
|
|
546
|
@classmethod
|
547
|
def get_prompts(cls):
|
548
|
rp = chirp_common.RadioPrompts()
|
549
|
|
550
|
rp.pre_download = _(
|
551
|
"1. Turn radio on.\n"
|
552
|
"2. Connect cable to DATA terminal.\n"
|
553
|
"3. Unclip battery.\n"
|
554
|
"4. Press and hold in the [AMS] key and power key while clipping"
|
555
|
" \n in back battery the"
|
556
|
"(\"ADMS\" will appear on the display).\n"
|
557
|
"5. <b>After clicking OK</b>, press the [BAND] key.\n")
|
558
|
rp.pre_upload = _(
|
559
|
"1. Turn radio on.\n"
|
560
|
"2. Connect cable to DATA terminal.\n"
|
561
|
"3. Unclip battery.\n"
|
562
|
"4. Press and hold in the [AMS] key and power key while clipping"
|
563
|
" \n in back battery the "
|
564
|
"(\"ADMS\" will appear on the display).\n"
|
565
|
"5. Press the [MODE] key (\"-WAIT-\" will appear on the LCD).\n"
|
566
|
"<b>Then click OK</b>")
|
567
|
return rp
|
568
|
|
569
|
def process_mmap(self):
|
570
|
|
571
|
mem_format = MEM_SETTINGS_FORMAT + MEM_FORMAT + MEM_CALLSIGN_FORMAT + MEM_CHECKSUM_FORMAT
|
572
|
|
573
|
self._memobj = bitwise.parse(mem_format % self._mem_params, self._mmap)
|
574
|
|
575
|
def get_features(self):
|
576
|
rf = chirp_common.RadioFeatures()
|
577
|
rf.has_dtcs_polarity = False
|
578
|
rf.valid_modes = list(MODES)
|
579
|
rf.valid_tmodes = list(TMODES)
|
580
|
rf.valid_duplexes = list(DUPLEX)
|
581
|
rf.valid_tuning_steps = list(STEPS)
|
582
|
rf.valid_bands = [(500000, 999900000)]
|
583
|
rf.valid_skips = SKIPS
|
584
|
rf.valid_power_levels = POWER_LEVELS
|
585
|
rf.valid_characters = "".join(CHARSET)
|
586
|
rf.valid_name_length = 6
|
587
|
rf.memory_bounds = (1, 900)
|
588
|
rf.can_odd_split = True
|
589
|
rf.has_ctone = False
|
590
|
rf.has_bank_names = True
|
591
|
rf.has_settings = True
|
592
|
return rf
|
593
|
|
594
|
def get_raw_memory(self, number):
|
595
|
return "\n".join([repr(self._memobj.memory[number - 1]),
|
596
|
repr(self._memobj.flag[number - 1])])
|
597
|
|
598
|
def _checksums(self):
|
599
|
return [yaesu_clone.YaesuChecksum(0x0000, 0xFEC9)] # The whole file -2 bytes
|
600
|
|
601
|
@staticmethod
|
602
|
def _add_ff_pad(val, length):
|
603
|
return val.ljust(length, "\xFF")[:length]
|
604
|
|
605
|
@classmethod
|
606
|
def _strip_ff_pads(cls, messages):
|
607
|
result = []
|
608
|
for msg_text in messages:
|
609
|
result.append(str(msg_text).rstrip("\xFF"))
|
610
|
return result
|
611
|
|
612
|
def get_memory(self, number):
|
613
|
flag = self._memobj.flag[number - 1]
|
614
|
_mem = self._memobj.memory[number - 1]
|
615
|
|
616
|
mem = chirp_common.Memory()
|
617
|
mem.number = number
|
618
|
if not flag.used:
|
619
|
mem.empty = True
|
620
|
if not flag.valid:
|
621
|
mem.empty = True
|
622
|
return mem
|
623
|
mem.freq = chirp_common.fix_rounded_step(int(_mem.freq) * 1000)
|
624
|
mem.offset = int(_mem.offset) * 1000
|
625
|
mem.rtone = mem.ctone = chirp_common.TONES[_mem.tone]
|
626
|
self._get_tmode(mem, _mem)
|
627
|
mem.duplex = DUPLEX[_mem.duplex]
|
628
|
if mem.duplex == "split":
|
629
|
mem.offset = chirp_common.fix_rounded_step(mem.offset)
|
630
|
mem.mode = self._decode_mode(_mem)
|
631
|
mem.dtcs = chirp_common.DTCS_CODES[_mem.dcs]
|
632
|
mem.tuning_step = STEPS[_mem.tune_step]
|
633
|
mem.power = self._decode_power_level(_mem)
|
634
|
mem.skip = flag.pskip and "P" or flag.skip and "S" or ""
|
635
|
mem.name = self._decode_label(_mem)
|
636
|
|
637
|
return mem
|
638
|
|
639
|
def _decode_label(self, mem):
|
640
|
return str(mem.label).rstrip("\xFF")
|
641
|
|
642
|
def _encode_label(self, mem):
|
643
|
return self._add_ff_pad(mem.name.rstrip(), 6)
|
644
|
|
645
|
def _encode_charsetbits(self, mem):
|
646
|
# We only speak English here in chirpville
|
647
|
return [0x00, 0x00]
|
648
|
|
649
|
def _decode_power_level(self, mem): # 3 High 2 Mid 1 Low
|
650
|
return POWER_LEVELS[3 - mem.power]
|
651
|
|
652
|
def _encode_power_level(self, mem):
|
653
|
return 3 - POWER_LEVELS.index(mem.power)
|
654
|
|
655
|
def _decode_mode(self, mem):
|
656
|
mode = MODES[mem.mode]
|
657
|
if mode == 'FM' and int(mem.deviation):
|
658
|
return 'NFM'
|
659
|
else:
|
660
|
return mode
|
661
|
|
662
|
def _encode_mode(self, mem):
|
663
|
mode = mem.mode
|
664
|
if mode == 'NFM':
|
665
|
# Narrow is handled by a separate flag
|
666
|
mode = 'FM'
|
667
|
return MODES.index(mode)
|
668
|
|
669
|
def _get_tmode(self, mem, _mem):
|
670
|
mem.tmode = TMODES[_mem.tone_mode]
|
671
|
|
672
|
def _set_tmode(self, _mem, mem):
|
673
|
_mem.tone_mode = TMODES.index(mem.tmode)
|
674
|
|
675
|
def _set_mode(self, _mem, mem):
|
676
|
_mem.deviation = mem.mode == 'NFM'
|
677
|
_mem.mode = self._encode_mode(mem)
|
678
|
|
679
|
def _debank(self, mem):
|
680
|
bm = self.get_bank_model()
|
681
|
for bank in bm.get_memory_mappings(mem):
|
682
|
bm.remove_memory_from_mapping(mem, bank)
|
683
|
|
684
|
def set_memory(self, mem):
|
685
|
_mem = self._memobj.memory[mem.number - 1]
|
686
|
flag = self._memobj.flag[mem.number - 1]
|
687
|
|
688
|
self._debank(mem)
|
689
|
|
690
|
if not mem.empty and not flag.valid:
|
691
|
self._wipe_memory(_mem)
|
692
|
|
693
|
if mem.empty and flag.valid and not flag.used:
|
694
|
flag.valid = False
|
695
|
return
|
696
|
flag.used = not mem.empty
|
697
|
flag.valid = flag.used
|
698
|
|
699
|
if mem.empty:
|
700
|
return
|
701
|
|
702
|
if mem.freq < 30000000 or \
|
703
|
(mem.freq > 88000000 and mem.freq < 108000000) or \
|
704
|
mem.freq > 580000000:
|
705
|
flag.nosubvfo = True # Masked from VFO B
|
706
|
else:
|
707
|
flag.nosubvfo = False # Available in both VFOs
|
708
|
|
709
|
_mem.freq = int(mem.freq / 1000)
|
710
|
_mem.offset = int(mem.offset / 1000)
|
711
|
_mem.tone = chirp_common.TONES.index(mem.rtone)
|
712
|
self._set_tmode(_mem, mem)
|
713
|
_mem.duplex = DUPLEX.index(mem.duplex)
|
714
|
self._set_mode(_mem, mem)
|
715
|
_mem.dcs = chirp_common.DTCS_CODES.index(mem.dtcs)
|
716
|
_mem.tune_step = STEPS.index(mem.tuning_step)
|
717
|
|
718
|
if mem.power:
|
719
|
_mem.power = self._encode_power_level(mem)
|
720
|
else:
|
721
|
_mem.power = 3 # Set 3 - High power as the default
|
722
|
|
723
|
_mem.label = self._encode_label(mem)
|
724
|
charsetbits = self._encode_charsetbits(mem)
|
725
|
_mem.charsetbits[0], _mem.charsetbits[1] = charsetbits
|
726
|
|
727
|
flag.skip = mem.skip == "S"
|
728
|
flag.pskip = mem.skip == "P"
|
729
|
|
730
|
_mem.display_tag = 1 # Always Display Memory Name (For the moment..)
|
731
|
|
732
|
@classmethod
|
733
|
def _wipe_memory(cls, mem):
|
734
|
mem.set_raw("\x00" * (mem.size() // 8))
|
735
|
mem.unknown1 = 0x05
|
736
|
|
737
|
def get_bank_model(self):
|
738
|
return FT70BankModel(self)
|
739
|
|
740
|
def _get_dtmf_settings(self):
|
741
|
menu = RadioSettingGroup("dtmf_settings", "DTMF")
|
742
|
dtmf = self._memobj.scan_settings
|
743
|
|
744
|
val = RadioSettingValueList(
|
745
|
self._DTMF_MODE,
|
746
|
self._DTMF_MODE[dtmf.dtmf_mode])
|
747
|
rs = RadioSetting("scan_settings.dtmf_mode", "DTMF Mode", val)
|
748
|
menu.append(rs)
|
749
|
|
750
|
val = RadioSettingValueList(
|
751
|
self._DTMF_DELAY,
|
752
|
self._DTMF_DELAY[dtmf.dtmf_delay])
|
753
|
rs = RadioSetting(
|
754
|
"scan_settings.dtmf_delay", "DTMF Delay", val)
|
755
|
menu.append(rs)
|
756
|
|
757
|
val = RadioSettingValueList(
|
758
|
self._DTMF_SPEED,
|
759
|
self._DTMF_SPEED[dtmf.dtmf_speed])
|
760
|
rs = RadioSetting(
|
761
|
"scan_settings.dtmf_speed", "DTMF Speed", val)
|
762
|
menu.append(rs)
|
763
|
|
764
|
for i in range(10):
|
765
|
|
766
|
name = "dtmf_%02d" % (i + 1)
|
767
|
if i == 9:
|
768
|
name = "dtmf_%02d" % 0
|
769
|
|
770
|
dtmfsetting = self._memobj.dtmf[i]
|
771
|
dtmfstr = ""
|
772
|
for c in dtmfsetting.memory:
|
773
|
if c == 0xFF:
|
774
|
break
|
775
|
if c < len(FT70_DTMF_CHARS):
|
776
|
dtmfstr += FT70_DTMF_CHARS[c]
|
777
|
dtmfentry = RadioSettingValueString(0, 16, dtmfstr)
|
778
|
dtmfentry.set_charset(
|
779
|
FT70_DTMF_CHARS + list("abcdef ")) # Allow input in lowercase, space ? validation fails otherwise
|
780
|
rs = RadioSetting(name, name.upper(), dtmfentry)
|
781
|
rs.set_apply_callback(self.apply_dtmf, i)
|
782
|
menu.append(rs)
|
783
|
|
784
|
return menu
|
785
|
|
786
|
def _get_display_settings(self):
|
787
|
menu = RadioSettingGroup("display_settings", "Display")
|
788
|
scan_settings = self._memobj.scan_settings
|
789
|
|
790
|
val = RadioSettingValueList(
|
791
|
self._LAMP_KEY,
|
792
|
self._LAMP_KEY[scan_settings.lamp])
|
793
|
rs = RadioSetting("scan_settings.lamp", "Lamp", val)
|
794
|
menu.append(rs)
|
795
|
|
796
|
val = RadioSettingValueList(
|
797
|
self._LCD_DIMMER,
|
798
|
self._LCD_DIMMER[scan_settings.lcd_dimmer])
|
799
|
rs = RadioSetting("scan_settings.lcd_dimmer", "LCD Dimmer", val)
|
800
|
menu.append(rs)
|
801
|
|
802
|
opening_message = self._memobj.opening_message
|
803
|
val = RadioSettingValueList(
|
804
|
self._OPENING_MESSAGE,
|
805
|
self._OPENING_MESSAGE[opening_message.flag])
|
806
|
rs = RadioSetting("opening_message.flag", "Opening Msg Mode", val)
|
807
|
menu.append(rs)
|
808
|
|
809
|
return menu
|
810
|
|
811
|
def _get_config_settings(self):
|
812
|
menu = RadioSettingGroup("config_settings", "Config")
|
813
|
scan_settings = self._memobj.scan_settings
|
814
|
|
815
|
# 02 APO Set the length of time until the transceiver turns off automatically.
|
816
|
|
817
|
first_settings = self._memobj.first_settings
|
818
|
val = RadioSettingValueList(
|
819
|
self._APO_SELECT,
|
820
|
self._APO_SELECT[first_settings.apo])
|
821
|
rs = RadioSetting("first_settings.apo", "APO", val)
|
822
|
menu.append(rs)
|
823
|
|
824
|
# 03 BCLO Turns the busy channel lockout function on/off.
|
825
|
|
826
|
val = RadioSettingValueList(
|
827
|
self._OFF_ON,
|
828
|
self._OFF_ON[scan_settings.bclo])
|
829
|
rs = RadioSetting("scan_settings.bclo", "Busy Channel Lockout", val)
|
830
|
menu.append(rs)
|
831
|
|
832
|
# 04 BEEP Sets the beep sound function.
|
833
|
|
834
|
beep_settings = self._memobj.beep_settings
|
835
|
val = RadioSettingValueList(
|
836
|
self._BEEP_SELECT,
|
837
|
self._BEEP_SELECT[beep_settings.beep_select])
|
838
|
rs = RadioSetting("beep_settings.beep_select", "Beep", val)
|
839
|
menu.append(rs)
|
840
|
|
841
|
# 05 BEP.LVL Beep volume setting LEVEL1 - LEVEL4 - LEVEL7
|
842
|
|
843
|
val = RadioSettingValueList(
|
844
|
self._BEEP_LEVEL,
|
845
|
self._BEEP_LEVEL[beep_settings.beep_level])
|
846
|
rs = RadioSetting("beep_settings", "Beep Level", val)
|
847
|
menu.append(rs)
|
848
|
|
849
|
# 06 BEP.Edg Sets the beep sound ON or OFF when a band edge is encountered.
|
850
|
|
851
|
val = RadioSettingValueList(
|
852
|
self._OFF_ON,
|
853
|
self._OFF_ON[scan_settings.beep_edge])
|
854
|
rs = RadioSetting("scan_settings.beep_edge", "Beep Band Edge", val)
|
855
|
menu.append(rs)
|
856
|
|
857
|
# 10 Bsy.LED Turn the MODE/STATUS Indicator ON or OFF while receiving signals.
|
858
|
|
859
|
val = RadioSettingValueList(
|
860
|
self._ON_OFF,
|
861
|
self._ON_OFF[scan_settings.busy_led])
|
862
|
rs = RadioSetting("scan_settings.busy_led", "Busy LED", val)
|
863
|
menu.append(rs)
|
864
|
|
865
|
# 26 HOME/REV Select the function of the [HOME/REV] key.
|
866
|
|
867
|
val = RadioSettingValueList(
|
868
|
self._HOME_REV,
|
869
|
self._HOME_REV[scan_settings.home_rev])
|
870
|
rs = RadioSetting("scan_settings.home_rev", "HOME/REV", val)
|
871
|
menu.append(rs)
|
872
|
|
873
|
# 27 HOME->VFO Turn transfer VFO to the Home channel ON or OFF.
|
874
|
|
875
|
val = RadioSettingValueList(
|
876
|
self._OFF_ON,
|
877
|
self._OFF_ON[scan_settings.home_vfo])
|
878
|
rs = RadioSetting("scan_settings.home_vfo", "Home->VFO", val)
|
879
|
menu.append(rs)
|
880
|
|
881
|
# 30 LOCK Configure the lock mode setting. KEY / DIAL / K+D / PTT / K+P / D+P / ALL
|
882
|
|
883
|
val = RadioSettingValueList(
|
884
|
self._LOCK,
|
885
|
self._LOCK[scan_settings.lock])
|
886
|
rs = RadioSetting("scan_settings.lock", "Lock Mode", val)
|
887
|
menu.append(rs)
|
888
|
|
889
|
# 32 Mon/T-Call Select the function of the [MONI/T-CALL] switch.
|
890
|
|
891
|
val = RadioSettingValueList(
|
892
|
self._MONI_TCALL,
|
893
|
self._MONI_TCALL[scan_settings.moni])
|
894
|
rs = RadioSetting("scan_settings.moni", "MONI/T-CALL", val)
|
895
|
menu.append(rs)
|
896
|
|
897
|
# 42 PTT.DLY Set the PTT delay time. OFF / 20 ms / 50 ms / 100 ms / 200 ms
|
898
|
|
899
|
val = RadioSettingValueList(
|
900
|
self._PTT_DELAY,
|
901
|
self._PTT_DELAY[scan_settings.ptt_delay])
|
902
|
rs = RadioSetting("scan_settings.ptt_delay", "PTT Delay", val)
|
903
|
menu.append(rs)
|
904
|
|
905
|
# 45 RPT.ARS Turn the ARS function on/off.
|
906
|
|
907
|
val = RadioSettingValueList(
|
908
|
self._OFF_ON,
|
909
|
self._OFF_ON[scan_settings.ars])
|
910
|
rs = RadioSetting("scan_settings.ars", "ARS", val)
|
911
|
menu.append(rs)
|
912
|
|
913
|
# 48 RX.SAVE Set the battery save time. OFF / 0.2 s - 60.0 s
|
914
|
|
915
|
val = RadioSettingValueList(
|
916
|
self._RX_SAVE,
|
917
|
self._RX_SAVE[scan_settings.rx_save])
|
918
|
rs = RadioSetting("scan_settings.rx_save", "RX SAVE", val)
|
919
|
menu.append(rs)
|
920
|
|
921
|
# 60 VFO.MOD Set the frequency setting range in the VFO mode by DIAL knob. ALL / BAND
|
922
|
|
923
|
val = RadioSettingValueList(
|
924
|
self._VFO_MODE,
|
925
|
self._VFO_MODE[scan_settings.vfo_mode])
|
926
|
rs = RadioSetting("scan_settings.vfo_mode", "VFO MODE", val)
|
927
|
menu.append(rs)
|
928
|
|
929
|
# 56 TOT Set the timeout timer.
|
930
|
|
931
|
val = RadioSettingValueList(
|
932
|
self._TOT_TIME,
|
933
|
self._TOT_TIME[scan_settings.tot])
|
934
|
rs = RadioSetting("scan_settings.tot", "Transmit Timeout (TOT)", val)
|
935
|
menu.append(rs)
|
936
|
|
937
|
# 31 MCGAIN Adjust the microphone gain level
|
938
|
|
939
|
val = RadioSettingValueList(
|
940
|
self._MIC_GAIN,
|
941
|
self._MIC_GAIN[scan_settings.mic_gain])
|
942
|
rs = RadioSetting("scan_settings.mic_gain", "Mic Gain", val)
|
943
|
menu.append(rs)
|
944
|
|
945
|
# VOLUME Adjust the volume level
|
946
|
|
947
|
scan_settings_2 = self._memobj.scan_settings_2
|
948
|
val = RadioSettingValueList(
|
949
|
self._VOLUME,
|
950
|
self._VOLUME[scan_settings_2.volume])
|
951
|
rs = RadioSetting("scan_settings_2.volume", "Volume", val)
|
952
|
menu.append(rs)
|
953
|
|
954
|
# Squelch F key, Hold Monitor, Dial to adjust squelch level
|
955
|
|
956
|
squelch_settings = self._memobj.squelch_settings
|
957
|
val = RadioSettingValueList(
|
958
|
self._SQUELCH,
|
959
|
self._SQUELCH[squelch_settings.squelch])
|
960
|
rs = RadioSetting("squelch_settings.squelch", "Squelch", val)
|
961
|
menu.append(rs)
|
962
|
|
963
|
return menu
|
964
|
|
965
|
def _get_digital_settings(self):
|
966
|
menu = RadioSettingGroup("digital_settings", "Digital")
|
967
|
|
968
|
# MYCALL
|
969
|
mycall = self._memobj.my_call
|
970
|
mycallstr = str(mycall.callsign).rstrip("\xFF")
|
971
|
|
972
|
mycallentry = RadioSettingValueString(0, 10, mycallstr, False, charset=self._MYCALL_CHR_SET)
|
973
|
rs = RadioSetting('mycall.callsign', 'MYCALL', mycallentry)
|
974
|
rs.set_apply_callback(self.apply_mycall, mycall)
|
975
|
menu.append(rs)
|
976
|
|
977
|
# Short Press AMS button AMS TX Mode
|
978
|
|
979
|
digital_settings = self._memobj.digital_settings
|
980
|
val = RadioSettingValueList(
|
981
|
self._AMS_TX_MODE,
|
982
|
self._AMS_TX_MODE[digital_settings.ams_tx_mode])
|
983
|
rs = RadioSetting("digital_settings.ams_tx_mode", "AMS TX Mode", val)
|
984
|
menu.append(rs)
|
985
|
|
986
|
# 16 DIG VW Turn the VW mode selection ON or OFF.
|
987
|
|
988
|
val = RadioSettingValueList(
|
989
|
self._VW_MODE,
|
990
|
self._VW_MODE[digital_settings.vw_mode])
|
991
|
rs = RadioSetting("digital_settings.vw_mode", "VW Mode", val)
|
992
|
menu.append(rs)
|
993
|
|
994
|
# TX DG-ID Long Press Mode Key, Dial
|
995
|
|
996
|
val = RadioSettingValueList(
|
997
|
self._DG_ID,
|
998
|
self._DG_ID[digital_settings.tx_dg_id])
|
999
|
rs = RadioSetting("digital_settings.tx_dg_id", "TX DG-ID", val)
|
1000
|
menu.append(rs)
|
1001
|
|
1002
|
# RX DG-ID Long Press Mode Key, Mode Key to select, Dial
|
1003
|
|
1004
|
val = RadioSettingValueList(
|
1005
|
self._DG_ID,
|
1006
|
self._DG_ID[digital_settings.rx_dg_id])
|
1007
|
rs = RadioSetting("digital_settings.rx_dg_id", "RX DG-ID", val)
|
1008
|
menu.append(rs)
|
1009
|
|
1010
|
# 15 DIG.POP Call sign display pop up time
|
1011
|
|
1012
|
# 00 OFF 00
|
1013
|
# 0A 2s 10
|
1014
|
# 0B 4s 11
|
1015
|
# 0C 6s 12
|
1016
|
# 0D 8s 13
|
1017
|
# 0E 10s 14
|
1018
|
# 0F 20s 15
|
1019
|
# 10 30s 16
|
1020
|
# 11 60s 17
|
1021
|
# 12 CONT 18
|
1022
|
|
1023
|
digital_settings_more = self._memobj.digital_settings_more
|
1024
|
|
1025
|
val = RadioSettingValueList(
|
1026
|
self._DIG_POP_UP,
|
1027
|
self._DIG_POP_UP[
|
1028
|
0 if digital_settings_more.digital_popup == 0 else digital_settings_more.digital_popup - 9])
|
1029
|
|
1030
|
rs = RadioSetting("digital_settings_more.digital_popup", "Digital Popup", val)
|
1031
|
rs.set_apply_callback(self.apply_digital_popup, digital_settings_more)
|
1032
|
menu.append(rs)
|
1033
|
|
1034
|
# 07 BEP.STB Standby Beep in the digital C4FM mode. On/Off
|
1035
|
|
1036
|
val = RadioSettingValueList(
|
1037
|
self._STANDBY_BEEP,
|
1038
|
self._STANDBY_BEEP[digital_settings.standby_beep])
|
1039
|
rs = RadioSetting("digital_settings.standby_beep", "Standby Beep", val)
|
1040
|
menu.append(rs)
|
1041
|
|
1042
|
return menu
|
1043
|
|
1044
|
def _get_gm_settings(self):
|
1045
|
menu = RadioSettingGroup("first_settings", "Group Monitor")
|
1046
|
|
1047
|
# 24 GM RNG Select the beep option while receiving digital GM information. OFF / IN RNG /ALWAYS
|
1048
|
|
1049
|
first_settings = self._memobj.first_settings
|
1050
|
val = RadioSettingValueList(
|
1051
|
self._GM_RING,
|
1052
|
self._GM_RING[first_settings.gm_ring])
|
1053
|
rs = RadioSetting("first_settings.gm_ring", "GM Ring", val)
|
1054
|
menu.append(rs)
|
1055
|
|
1056
|
# 25 GM INT Set the transmission interval of digital GM information. OFF / NORMAL / LONG
|
1057
|
|
1058
|
scan_settings = self._memobj.scan_settings
|
1059
|
val = RadioSettingValueList(
|
1060
|
self._GM_INTERVAL,
|
1061
|
self._GM_INTERVAL[scan_settings.gm_interval])
|
1062
|
rs = RadioSetting("scan_settings.gm_interval", "GM Interval", val)
|
1063
|
menu.append(rs)
|
1064
|
|
1065
|
return menu
|
1066
|
|
1067
|
def _get_scan_settings(self):
|
1068
|
menu = RadioSettingGroup("scan_settings", "Scan")
|
1069
|
scan_settings = self._memobj.scan_settings
|
1070
|
|
1071
|
# 23 DW RVT Turn the "Priority Channel Revert" feature ON or OFF during Dual Receive.
|
1072
|
|
1073
|
val = RadioSettingValueList(
|
1074
|
self._OFF_ON,
|
1075
|
self._OFF_ON[scan_settings.dw_rt])
|
1076
|
rs = RadioSetting("scan_settings.dw_rt", "Dual Watch Priority Channel Revert", val)
|
1077
|
menu.append(rs)
|
1078
|
|
1079
|
# 21 DW INT Set the priority memory channel monitoring interval during Dual Receive. 0.1S - 5.0S - 10.0S
|
1080
|
|
1081
|
val = RadioSettingValueList(
|
1082
|
self._SCAN_RESTART,
|
1083
|
self._SCAN_RESTART[scan_settings.dw_interval])
|
1084
|
rs = RadioSetting("scan_settings.dw_interval", "Dual Watch Interval", val)
|
1085
|
menu.append(rs)
|
1086
|
|
1087
|
# 22 DW RSM Configure the scan stop mode settings for Dual Receive. 2.0S - 10.0 S / BUSY / HOLD
|
1088
|
|
1089
|
first_settings = self._memobj.first_settings
|
1090
|
val = RadioSettingValueList(
|
1091
|
self._SCAN_RESUME,
|
1092
|
self._SCAN_RESUME[first_settings.dw_resume_interval])
|
1093
|
rs = RadioSetting("first_settings.dw_resume_interval", "Dual Watch Resume Interval", val)
|
1094
|
menu.append(rs)
|
1095
|
|
1096
|
# 51 SCN.LMP Set the scan lamp ON or OFF when scanning stops. OFF / ON
|
1097
|
|
1098
|
val = RadioSettingValueList(
|
1099
|
self._OFF_ON,
|
1100
|
self._OFF_ON[scan_settings.scan_lamp])
|
1101
|
rs = RadioSetting("scan_settings.scan_lamp", "Scan Lamp", val)
|
1102
|
menu.append(rs)
|
1103
|
|
1104
|
# 53 SCN.STR Set the scanning restart time. 0.1 S - 2.0 S - 10.0 S
|
1105
|
|
1106
|
val = RadioSettingValueList(
|
1107
|
self._SCAN_RESTART,
|
1108
|
self._SCAN_RESTART[scan_settings.scan_restart])
|
1109
|
rs = RadioSetting("scan_settings.scan_restart", "Scan Restart", val)
|
1110
|
menu.append(rs)
|
1111
|
|
1112
|
# Scan Width Section
|
1113
|
|
1114
|
# 50 SCV.WTH Set the VFO scan frequency range. BAND / ALL - NOT FOUND!
|
1115
|
|
1116
|
# Scan Resume Section
|
1117
|
|
1118
|
# 52 SCN.RSM Configure the scan stop mode settings. 2.0 S - 5.0 S - 10.0 S / BUSY / HOLD
|
1119
|
|
1120
|
first_settings = self._memobj.first_settings
|
1121
|
val = RadioSettingValueList(
|
1122
|
self._SCAN_RESUME,
|
1123
|
self._SCAN_RESUME[first_settings.scan_resume])
|
1124
|
rs = RadioSetting("first_settings.scan_resume", "Scan Resume", val)
|
1125
|
menu.append(rs)
|
1126
|
|
1127
|
return menu
|
1128
|
|
1129
|
def _get_settings(self):
|
1130
|
top = RadioSettings(
|
1131
|
self._get_config_settings(),
|
1132
|
self._get_digital_settings(),
|
1133
|
self._get_display_settings(),
|
1134
|
self._get_dtmf_settings(),
|
1135
|
self._get_gm_settings(),
|
1136
|
self._get_scan_settings()
|
1137
|
)
|
1138
|
return top
|
1139
|
|
1140
|
def get_settings(self):
|
1141
|
try:
|
1142
|
return self._get_settings()
|
1143
|
except:
|
1144
|
import traceback
|
1145
|
LOG.error("Failed to parse settings: %s", traceback.format_exc())
|
1146
|
return None
|
1147
|
|
1148
|
@classmethod
|
1149
|
def apply_ff_padded_string(cls, setting, obj):
|
1150
|
setattr(obj, "padded_string", cls._add_ff_pad(setting.value.get_value().rstrip(), 6))
|
1151
|
|
1152
|
def set_settings(self, settings):
|
1153
|
_mem = self._memobj
|
1154
|
for element in settings:
|
1155
|
if not isinstance(element, RadioSetting):
|
1156
|
self.set_settings(element)
|
1157
|
continue
|
1158
|
if not element.changed():
|
1159
|
continue
|
1160
|
try:
|
1161
|
if element.has_apply_callback():
|
1162
|
LOG.debug("Using apply callback")
|
1163
|
try:
|
1164
|
element.run_apply_callback()
|
1165
|
except NotImplementedError as e:
|
1166
|
LOG.error(e)
|
1167
|
continue
|
1168
|
|
1169
|
# Find the object containing setting.
|
1170
|
obj = _mem
|
1171
|
bits = element.get_name().split(".")
|
1172
|
setting = bits[-1]
|
1173
|
for name in bits[:-1]:
|
1174
|
if name.endswith("]"):
|
1175
|
name, index = name.split("[")
|
1176
|
index = int(index[:-1])
|
1177
|
obj = getattr(obj, name)[index]
|
1178
|
else:
|
1179
|
obj = getattr(obj, name)
|
1180
|
|
1181
|
try:
|
1182
|
old_val = getattr(obj, setting)
|
1183
|
LOG.debug("Setting %s(%r) <= %s" % (
|
1184
|
element.get_name(), old_val, element.value))
|
1185
|
setattr(obj, setting, element.value)
|
1186
|
except AttributeError as e:
|
1187
|
LOG.error("Setting %s is not in the memory map: %s" %
|
1188
|
(element.get_name(), e))
|
1189
|
except Exception:
|
1190
|
LOG.debug(element.get_name())
|
1191
|
raise
|
1192
|
|
1193
|
def apply_volume(cls, setting, vfo):
|
1194
|
val = setting.value.get_value()
|
1195
|
cls._memobj.vfo_info[(vfo * 2)].volume = val
|
1196
|
cls._memobj.vfo_info[(vfo * 2) + 1].volume = val
|
1197
|
|
1198
|
def apply_dtmf(cls, setting, i):
|
1199
|
rawval = setting.value.get_value().upper().rstrip()
|
1200
|
val = [FT70_DTMF_CHARS.index(x) for x in rawval]
|
1201
|
for x in range(len(val), 16):
|
1202
|
val.append(0xFF)
|
1203
|
cls._memobj.dtmf[i].memory = val
|
1204
|
|
1205
|
def apply_digital_popup(cls, setting, obj):
|
1206
|
rawval = setting.value.get_value()
|
1207
|
val = 0 if cls._DIG_POP_UP.index(rawval) == 0 else cls._DIG_POP_UP.index(rawval) + 9
|
1208
|
obj.digital_popup = val
|
1209
|
|
1210
|
def apply_mycall(cls, setting, obj):
|
1211
|
cs = setting.value.get_value()
|
1212
|
if cs[0] in ('-', '/'):
|
1213
|
raise InvalidValueError("First character of call sign can't be - or /: {0:s}".format(cs))
|
1214
|
else:
|
1215
|
obj.callsign = cls._add_ff_pad(cs.rstrip(), 10)
|
1216
|
|
1217
|
def load_mmap(self, filename):
|
1218
|
if filename.lower().endswith(self._adms_ext):
|
1219
|
with open(filename, 'rb') as f:
|
1220
|
self._adms_header = f.read(0xED)
|
1221
|
if b'=ADMS10, Version=1.0.1.0' not in self._adms_header:
|
1222
|
raise errors.ImageDetectFailed(
|
1223
|
'Unsupported version found in ADMS file')
|
1224
|
LOG.debug('ADMS Header:\n%s',
|
1225
|
util.hexprint(self._adms_header))
|
1226
|
self._mmap = memmap.MemoryMapBytes(self._model + f.read())
|
1227
|
LOG.info('Loaded ADMS file')
|
1228
|
self.process_mmap()
|
1229
|
else:
|
1230
|
chirp_common.CloneModeRadio.load_mmap(self, filename)
|
1231
|
|
1232
|
def save_mmap(self, filename):
|
1233
|
if filename.lower().endswith(self._adms_ext):
|
1234
|
if not hasattr(self, '_adms_header'):
|
1235
|
raise Exception('Unable to save .img to %s' % self._adms_ext)
|
1236
|
with open(filename, 'wb') as f:
|
1237
|
f.write(self._adms_header)
|
1238
|
f.write(self._mmap.get_packed()[5:])
|
1239
|
LOG.info('Wrote file')
|
1240
|
else:
|
1241
|
chirp_common.CloneModeRadio.save_mmap(self, filename)
|