Project

General

Profile

New Model #4145 » retevis_rt22.py

test driver module - please provide feedback - Jim Unroe, 11/12/2016 03:57 PM

 
1
# Copyright 2016 Jim Unroe <rock.unroe@gmail.com>
2
#
3
# This program is free software: you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation, either version 2 of the License, or
6
# (at your option) any later version.
7
#
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
# GNU General Public License for more details.
12
#
13
# You should have received a copy of the GNU General Public License
14
# along with this program.  If not, see <http://www.gnu.org/licenses/>.
15

    
16
import time
17
import os
18
import struct
19
import logging
20

    
21
from chirp import chirp_common, directory, memmap
22
from chirp import bitwise, errors, util
23
from chirp.settings import RadioSetting, RadioSettingGroup, \
24
    RadioSettingValueInteger, RadioSettingValueList, \
25
    RadioSettingValueBoolean, RadioSettings
26

    
27
LOG = logging.getLogger(__name__)
28

    
29
MEM_FORMAT = """
30
#seekto 0x0010;
31
struct {
32
  lbcd rxfreq[4];                 // confirmed
33
  lbcd txfreq[4];                 // confirmed
34
  ul16 rx_tone;                   // confirmed
35
  ul16 tx_tone;                   // confirmed
36
  u8 unknown1;
37
  u8 unknown3:2,
38
     highpower:1, // Power Level  // confirmed
39
     wide:1,      // Bandwidth    // confirmed
40
     unknown4:4;
41
  u8 unknown5[2];
42
} memory[16];
43

    
44
#seekto 0x012F;
45
struct {
46
  u8 voice;       // Voice Annunciation
47
  u8 tot;         // Time-out Timer                   
48
  u8 unknown1[3];
49
  u8 squelch;     // Squelch Level                    
50
  u8 save;        // Battery Saver                    
51
  u8 beep;        // Beep                             
52
  u8 unknown2[2];
53
  u8 vox;         // VOX                              
54
  u8 voxgain;     // VOX Gain                         
55
  u8 voxdelay;    // VOX Delay                        
56
  u8 unknown3[2];
57
  u8 pf2key;      // PF2 Key                          
58
} settings;
59

    
60
#seekto 0x017E;
61
u8 skipflags[2];  // SCAN_ADD
62
"""
63

    
64
CMD_ACK = "\x06"
65
RECV_BLOCK_SIZE = 0x40
66
SEND_BLOCK_SIZE = 0x10
67

    
68
RT22_POWER_LEVELS = [chirp_common.PowerLevel("Low",  watts=2.00),
69
                     chirp_common.PowerLevel("High", watts=5.00)]
70

    
71
RT22_DTCS = sorted(chirp_common.DTCS_CODES + [645])
72

    
73
PF2KEY_LIST = ["Scan", "Local Alarm", "Remote Alarm"]
74
TIMEOUTTIMER_LIST = [""] + ["%s seconds" % x for x in range(15, 615, 15)]
75
VOICE_LIST = ["Off", "Chinese", "English"]
76
VOX_LIST = ["OFF"] + ["%s" % x for x in range(1, 17)]
77
VOXDELAY_LIST = ["0.5", "1.0", "1.5", "2.0", "2.5", "3.0"]
78

    
79
SETTING_LISTS = {
80
    "pf2key": PF2KEY_LIST,
81
    "tot": TIMEOUTTIMER_LIST,
82
    "voice": VOICE_LIST,
83
    "vox": VOX_LIST,
84
    "voxdelay": VOXDELAY_LIST,
85
    }
86

    
87

    
88
def _rt22_enter_programming_mode(radio):
89
    serial = radio.pipe
90

    
91
    magic = "PROGRGS"
92
    exito = False
93
    for i in range(0, 5):
94
        for j in range(0, len(magic)):
95
            time.sleep(0.005)
96
            serial.write(magic[j])
97

    
98
        ack = serial.read(1)
99

    
100
        try:
101
            if ack == CMD_ACK:
102
                exito = True
103
                break
104
        except:
105
            LOG.debug("Attempt #%s, failed, trying again" % i)
106
            pass
107

    
108
    # check if we had EXITO
109
    if exito is False:
110
        msg = "The radio did not accept program mode after five tries.\n"
111
        msg += "Check you interface cable and power cycle your radio."
112
        raise errors.RadioError(msg)
113

    
114
    try:
115
        serial.write("\x02")
116
        ident = serial.read(8)
117
    except:
118
        raise errors.RadioError("Error communicating with radio")
119

    
120
    if not ident.startswith("P32073"):
121
        LOG.debug(util.hexprint(ident))
122
        raise errors.RadioError("Radio returned unknown identification string")
123

    
124
    try:
125
        serial.write(CMD_ACK)
126
        ack = serial.read(1)
127
    except:
128
        raise errors.RadioError("Error communicating with radio")
129

    
130
    if ack != CMD_ACK:
131
        raise errors.RadioError("Radio refused to enter programming mode")
132

    
133
    try:
134
        serial.write("\x07")
135
        ack = serial.read(1)
136
    except:
137
        raise errors.RadioError("Error communicating with radio")
138

    
139
    if ack != "\x4E":
140
        raise errors.RadioError("Radio refused to enter programming mode")
141

    
142

    
143
def _rt22_exit_programming_mode(radio):
144
    serial = radio.pipe
145
    try:
146
        serial.write("E")
147
    except:
148
        raise errors.RadioError("Radio refused to exit programming mode")
149

    
150

    
151
def _rt22_read_block(radio, block_addr, block_size):
152
    serial = radio.pipe
153

    
154
    cmd = struct.pack(">cHb", 'R', block_addr, RECV_BLOCK_SIZE)
155
    expectedresponse = "W" + cmd[1:]
156
    LOG.debug("Reading block %04x..." % (block_addr))
157

    
158
    try:
159
        for j in range(0, len(cmd)):
160
            time.sleep(0.005)
161
            serial.write(cmd[j])
162

    
163
        response = serial.read(4 + RECV_BLOCK_SIZE)
164
        if response[:4] != expectedresponse:
165
            raise Exception("Error reading block %04x." % (block_addr))
166

    
167
        block_data = response[4:]
168

    
169
        serial.write(CMD_ACK)
170
        ack = serial.read(1)
171
    except:
172
        raise errors.RadioError("Failed to read block at %04x" % block_addr)
173

    
174
    if ack != CMD_ACK:
175
        raise Exception("No ACK reading block %04x." % (block_addr))
176

    
177
    return block_data
178

    
179

    
180
def _rt22_write_block(radio, block_addr, block_size):
181
    serial = radio.pipe
182

    
183
    cmd = struct.pack(">cHb", 'W', block_addr, SEND_BLOCK_SIZE)
184
    data = radio.get_mmap()[block_addr:block_addr + SEND_BLOCK_SIZE]
185

    
186
    LOG.debug("Writing Data:")
187
    LOG.debug(util.hexprint(cmd + data))
188

    
189
    try:
190
        for j in range(0, len(cmd)):
191
            time.sleep(0.01)
192
            serial.write(cmd[j])
193
        for j in range(0, len(data)):
194
            time.sleep(0.01)
195
            serial.write(data[j])
196
        if serial.read(1) != CMD_ACK:
197
            raise Exception("No ACK")
198
    except:
199
        raise errors.RadioError("Failed to send block "
200
                                "to radio at %04x" % block_addr)
201

    
202

    
203
def do_download(radio):
204
    LOG.debug("download")
205
    _rt22_enter_programming_mode(radio)
206

    
207
    data = ""
208

    
209
    status = chirp_common.Status()
210
    status.msg = "Cloning from radio"
211

    
212
    status.cur = 0
213
    status.max = radio._memsize
214

    
215
    for addr in range(0, radio._memsize, RECV_BLOCK_SIZE):
216
        status.cur = addr + RECV_BLOCK_SIZE
217
        radio.status_fn(status)
218

    
219
        block = _rt22_read_block(radio, addr, RECV_BLOCK_SIZE)
220
        data += block
221

    
222
        LOG.debug("Address: %04x" % addr)
223
        LOG.debug(util.hexprint(block))
224

    
225
    data += radio.MODEL.ljust(8)
226

    
227
    _rt22_exit_programming_mode(radio)
228

    
229
    return memmap.MemoryMap(data)
230

    
231

    
232
def do_upload(radio):
233
    status = chirp_common.Status()
234
    status.msg = "Uploading to radio"
235

    
236
    _rt22_enter_programming_mode(radio)
237

    
238
    status.cur = 0
239
    status.max = radio._memsize
240

    
241
    for start_addr, end_addr in radio._ranges:
242
        for addr in range(start_addr, end_addr, SEND_BLOCK_SIZE):
243
            status.cur = addr + SEND_BLOCK_SIZE
244
            radio.status_fn(status)
245
            _rt22_write_block(radio, addr, SEND_BLOCK_SIZE)
246

    
247
    _rt22_exit_programming_mode(radio)
248

    
249

    
250
def model_match(cls, data):
251
    """Match the opened/downloaded image to the correct version"""
252

    
253
    if len(data) == 0x0348:
254
        rid = data[0x0340:0x0348]
255
        return rid.startswith(cls.MODEL)
256
    else:
257
        return False
258

    
259

    
260
@directory.register
261
class RT22Radio(chirp_common.CloneModeRadio):
262
    """Retevis RT22"""
263
    VENDOR = "Retevis"
264
    MODEL = "RT22"
265
    BAUD_RATE = 9600
266

    
267
    _ranges = [
268
               (0x0000, 0x0340),
269
              ]
270
    _memsize = 0x0340
271

    
272
    def get_features(self):
273
        rf = chirp_common.RadioFeatures()
274
        rf.has_settings = True
275
        rf.has_bank = False
276
        rf.has_ctone = True
277
        rf.has_cross = True
278
        rf.has_rx_dtcs = True
279
        rf.has_tuning_step = False
280
        rf.can_odd_split = True
281
        rf.has_name = False
282
        rf.valid_skips = ["", "S"]
283
        rf.valid_tmodes = ["", "Tone", "TSQL", "DTCS", "Cross"]
284
        rf.valid_cross_modes = ["Tone->Tone", "Tone->DTCS", "DTCS->Tone",
285
                                "->Tone", "->DTCS", "DTCS->", "DTCS->DTCS"]
286
        rf.valid_power_levels = RT22_POWER_LEVELS
287
        rf.valid_duplexes = ["", "-", "+", "split", "off"]
288
        rf.valid_modes = ["NFM", "FM"]  # 12.5 KHz, 25 kHz.
289
        rf.memory_bounds = (1, 16)
290
        rf.valid_bands = [(400000000, 520000000)]
291

    
292
        return rf
293

    
294
    def process_mmap(self):
295
        self._memobj = bitwise.parse(MEM_FORMAT, self._mmap)
296

    
297
    def sync_in(self):
298
        self._mmap = do_download(self)
299
        self.process_mmap()
300

    
301
    def sync_out(self):
302
        do_upload(self)
303

    
304
    def get_raw_memory(self, number):
305
        return repr(self._memobj.memory[number - 1])
306

    
307
    def _get_tone(self, _mem, mem):
308
        def _get_dcs(val):
309
            code = int("%03o" % (val & 0x07FF))
310
            pol = (val & 0x8000) and "R" or "N"
311
            return code, pol
312

    
313
        if _mem.tx_tone != 0xFFFF and _mem.tx_tone > 0x2800:
314
            tcode, tpol = _get_dcs(_mem.tx_tone)
315
            mem.dtcs = tcode
316
            txmode = "DTCS"
317
        elif _mem.tx_tone != 0xFFFF:
318
            mem.rtone = _mem.tx_tone / 10.0
319
            txmode = "Tone"
320
        else:
321
            txmode = ""
322

    
323
        if _mem.rx_tone != 0xFFFF and _mem.rx_tone > 0x2800:
324
            rcode, rpol = _get_dcs(_mem.rx_tone)
325
            mem.rx_dtcs = rcode
326
            rxmode = "DTCS"
327
        elif _mem.rx_tone != 0xFFFF:
328
            mem.ctone = _mem.rx_tone / 10.0
329
            rxmode = "Tone"
330
        else:
331
            rxmode = ""
332

    
333
        if txmode == "Tone" and not rxmode:
334
            mem.tmode = "Tone"
335
        elif txmode == rxmode and txmode == "Tone" and mem.rtone == mem.ctone:
336
            mem.tmode = "TSQL"
337
        elif txmode == rxmode and txmode == "DTCS" and mem.dtcs == mem.rx_dtcs:
338
            mem.tmode = "DTCS"
339
        elif rxmode or txmode:
340
            mem.tmode = "Cross"
341
            mem.cross_mode = "%s->%s" % (txmode, rxmode)
342

    
343
        if mem.tmode == "DTCS":
344
            mem.dtcs_polarity = "%s%s" % (tpol, rpol)
345

    
346
        LOG.debug("Got TX %s (%i) RX %s (%i)" %
347
                  (txmode, _mem.tx_tone, rxmode, _mem.rx_tone))
348

    
349
    def get_memory(self, number):
350
        bitpos = (1 << ((number - 1) % 8))
351
        bytepos = ((number - 1) / 8)
352
        LOG.debug("bitpos %s" % bitpos)
353
        LOG.debug("bytepos %s" % bytepos)
354

    
355
        _mem = self._memobj.memory[number - 1]
356
        _skp = self._memobj.skipflags[bytepos]
357

    
358
        mem = chirp_common.Memory()
359

    
360
        mem.number = number
361
        mem.freq = int(_mem.rxfreq) * 10
362

    
363
        # We'll consider any blank (i.e. 0MHz frequency) to be empty
364
        if mem.freq == 0:
365
            mem.empty = True
366
            return mem
367

    
368
        if _mem.rxfreq.get_raw() == "\xFF\xFF\xFF\xFF":
369
            mem.freq = 0
370
            mem.empty = True
371
            return mem
372

    
373
        if int(_mem.rxfreq) == int(_mem.txfreq):
374
            mem.duplex = ""
375
            mem.offset = 0
376
        else:
377
            mem.duplex = int(_mem.rxfreq) > int(_mem.txfreq) and "-" or "+"
378
            mem.offset = abs(int(_mem.rxfreq) - int(_mem.txfreq)) * 10
379

    
380
        mem.mode = _mem.wide and "FM" or "NFM"
381

    
382
        self._get_tone(_mem, mem)
383

    
384
        mem.power = RT22_POWER_LEVELS[_mem.highpower]
385

    
386
        mem.skip = "" if (_skp & bitpos) else "S"
387
        LOG.debug("mem.skip %s" % mem.skip)
388

    
389
        return mem
390

    
391
    def _set_tone(self, mem, _mem):
392
        def _set_dcs(code, pol):
393
            val = int("%i" % code, 8) + 0x2800
394
            if pol == "R":
395
                val += 0x8000
396
            return val
397

    
398
        if mem.tmode == "Cross":
399
            tx_mode, rx_mode = mem.cross_mode.split("->")
400
        elif mem.tmode == "Tone":
401
            tx_mode = mem.tmode
402
            rx_mode = None
403
        else:
404
            tx_mode = rx_mode = mem.tmode
405

    
406
        if tx_mode == "DTCS":
407
            _mem.tx_tone = mem.tmode != "DTCS" and \
408
                           _set_dcs(mem.dtcs, mem.dtcs_polarity[0]) or \
409
                           _set_dcs(mem.rx_dtcs, mem.dtcs_polarity[0])
410
        elif tx_mode:
411
            _mem.tx_tone = tx_mode == "Tone" and \
412
                int(mem.rtone * 10) or int(mem.ctone * 10)
413
        else:
414
            _mem.tx_tone = 0xFFFF
415

    
416
        if rx_mode == "DTCS":
417
            _mem.rx_tone = _set_dcs(mem.rx_dtcs, mem.dtcs_polarity[1])
418
        elif rx_mode:
419
            _mem.rx_tone = int(mem.ctone * 10)
420
        else:
421
            _mem.rx_tone = 0xFFFF
422

    
423
        LOG.debug("Set TX %s (%i) RX %s (%i)" %
424
                  (tx_mode, _mem.tx_tone, rx_mode, _mem.rx_tone))
425

    
426
    def set_memory(self, mem):
427
        bitpos = (1 << ((mem.number - 1) % 8))
428
        bytepos = ((mem.number - 1) / 8)
429
        LOG.debug("bitpos %s" % bitpos)
430
        LOG.debug("bytepos %s" % bytepos)
431

    
432
        _mem = self._memobj.memory[mem.number - 1]
433
        _skp = self._memobj.skipflags[bytepos]
434

    
435
        if mem.empty:
436
            _mem.set_raw("\xFF" * (_mem.size() / 8))
437
            return
438

    
439
        _mem.rxfreq = mem.freq / 10
440

    
441
        if mem.duplex == "off":
442
            for i in range(0, 4):
443
                _mem.txfreq[i].set_raw("\xFF")
444
        elif mem.duplex == "split":
445
            _mem.txfreq = mem.offset / 10
446
        elif mem.duplex == "+":
447
            _mem.txfreq = (mem.freq + mem.offset) / 10
448
        elif mem.duplex == "-":
449
            _mem.txfreq = (mem.freq - mem.offset) / 10
450
        else:
451
            _mem.txfreq = mem.freq / 10
452

    
453
        _mem.wide = mem.mode == "FM"
454

    
455
        self._set_tone(mem, _mem)
456

    
457
        _mem.highpower = mem.power == RT22_POWER_LEVELS[1]
458

    
459
        if mem.skip != "S":
460
            _skp |= bitpos
461
        else:
462
            _skp &= ~bitpos
463
        LOG.debug("_skp %s" % _skp)
464

    
465
    def get_settings(self):
466
        _settings = self._memobj.settings
467
        basic = RadioSettingGroup("basic", "Basic Settings")
468
        top = RadioSettings(basic)
469

    
470
        rs = RadioSetting("squelch", "Squelch Level",
471
                          RadioSettingValueInteger(0, 9, _settings.squelch))
472
        basic.append(rs)
473

    
474
        rs = RadioSetting("tot", "Time-out timer",
475
                          RadioSettingValueList(
476
                              TIMEOUTTIMER_LIST,
477
                              TIMEOUTTIMER_LIST[_settings.tot]))
478
        basic.append(rs)
479

    
480
        rs = RadioSetting("voice", "Voice Prompts",
481
                          RadioSettingValueList(
482
                              VOICE_LIST, VOICE_LIST[_settings.voice]))
483
        basic.append(rs)
484

    
485
        rs = RadioSetting("pf2key", "PF2 Key",
486
                          RadioSettingValueList(
487
                              PF2KEY_LIST, PF2KEY_LIST[_settings.pf2key]))
488
        basic.append(rs)
489

    
490
        rs = RadioSetting("vox", "Vox",
491
                          RadioSettingValueBoolean(_settings.vox))
492
        basic.append(rs)
493

    
494
        rs = RadioSetting("voxgain", "VOX Level",
495
                          RadioSettingValueList(
496
                              VOX_LIST, VOX_LIST[_settings.voxgain]))
497
        basic.append(rs)
498

    
499
        rs = RadioSetting("voxdelay", "VOX Delay Time",
500
                          RadioSettingValueList(
501
                              VOXDELAY_LIST,
502
                              VOXDELAY_LIST[_settings.voxdelay]))
503
        basic.append(rs)
504

    
505
        rs = RadioSetting("save", "Battery Save",
506
                          RadioSettingValueBoolean(_settings.save))
507
        basic.append(rs)
508

    
509
        rs = RadioSetting("beep", "Beep",
510
                          RadioSettingValueBoolean(_settings.beep))
511
        basic.append(rs)
512

    
513
        return top
514

    
515
    def set_settings(self, settings):
516
        for element in settings:
517
            if not isinstance(element, RadioSetting):
518
                self.set_settings(element)
519
                continue
520
            else:
521
                try:
522
                    if "." in element.get_name():
523
                        bits = element.get_name().split(".")
524
                        obj = self._memobj
525
                        for bit in bits[:-1]:
526
                            obj = getattr(obj, bit)
527
                        setting = bits[-1]
528
                    else:
529
                        obj = self._memobj.settings
530
                        setting = element.get_name()
531

    
532
                    LOG.debug("Setting %s = %s" % (setting, element.value))
533
                    setattr(obj, setting, element.value)
534
                except Exception, e:
535
                    LOG.debug(element.get_name())
536
                    raise
537

    
538
    @classmethod
539
    def match_model(cls, filedata, filename):
540
        match_size = False
541
        match_model = False
542

    
543
        # testing the file data size
544
        if len(filedata) in [0x0340, 0x0348]:
545
            match_size = True
546
        
547
        print "Match Size"
548
        print match_size
549

    
550
        # testing the firmware model fingerprint
551
        match_model = model_match(cls, filedata)
552

    
553
        print "Match Model"
554
        print match_model
555

    
556
        if match_size and match_model:
557
            return True
558
        else:
559
            return False
560

    
561
@directory.register
562
class KDC1(RT22Radio):
563
    """WLN KD-C1"""
564
    VENDOR = "WLN"
565
    MODEL = "KD-C1"
566

    
567
@directory.register
568
class ZTX6(RT22Radio):
569
    """Zastone ZT-X6"""
570
    VENDOR = "Zastone"
571
    MODEL = "ZT-X6"
(5-5/19)