Project

General

Profile

Bug #7187 » ic2820.py

Hakan Sengoltz, 11/11/2019 08:06 AM

 
1
# Copyright 2010 Dan Smith <dsmith@danplanet.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 3 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
from chirp.drivers import icf
17
from chirp import chirp_common, util, directory, bitwise
18

    
19
MEM_FORMAT = """
20
struct {
21
  u32  freq;
22
  u32  offset;
23
  char urcall[8];
24
  char r1call[8];
25
  char r2call[8];
26
  u8   unknown1;
27
  u8   unknown2:1,
28
       duplex:2,
29
       tmode:3,
30
       unknown3:2;
31
  u16  ctone:6,
32
       rtone:6,
33
       tune_step:4;
34
  u16  dtcs:7,
35
       mode:3,
36
       unknown4:6;
37
  u8   unknown5:1,
38
       digital_code:7;
39
  u8   unknown6:2,
40
       dtcs_polarity:2,
41
       unknown7:4;
42
  char name[8];
43
} memory[522];
44

    
45
#seekto 0x61E0;
46
u8 used_flags[66];
47

    
48
#seekto 0x6222;
49
u8 skip_flags[65];
50
u8 pskip_flags[65];
51

    
52
#seekto 0x62A4;
53
struct {
54
  u8 bank;
55
  u8 index;
56
} bank_info[500];
57

    
58
#seekto 0x66C0;
59
struct {
60
  char name[8];
61
} bank_names[26];
62

    
63
#seekto 0x6970;
64
struct {
65
  char call[8];
66
  u8 unknown[4];
67
} mycall[6];
68

    
69
#seekto 0x69B8;
70
struct {
71
  char call[8];
72
} urcall[60];
73

    
74
struct {
75
  char call[8];
76
} rptcall[60];
77

    
78
"""
79

    
80
TMODES = ["", "Tone", "??0", "TSQL", "TSQL-R", "??1", "DTCS", "DTCS-R"]
81
DUPLEX = ["", "-", "+", "+"]  # Not sure about index 3
82
MODES = ["FM", "NFM", "AM", "NAM", "DV"]
83
DTCSP = ["NN", "NR", "RN", "RR"]
84

    
85
MEM_LOC_SIZE = 48
86

    
87

    
88
class IC2820Bank(icf.IcomNamedBank):
89
    """An IC2820 bank"""
90
    def get_name(self):
91
        _banks = self._model._radio._memobj.bank_names
92
        return str(_banks[self.index].name).rstrip()
93

    
94
    def set_name(self, name):
95
        _banks = self._model._radio._memobj.bank_names
96
        _banks[self.index].name = str(name).ljust(8)[:8]
97

    
98

    
99
def _get_special():
100
    special = {"C0": 500 + 20,
101
               "C1": 500 + 21}
102

    
103
    for i in range(0, 10):
104
        ida = "%iA" % i
105
        idb = "%iB" % i
106
        special[ida] = 500 + i * 2
107
        special[idb] = 500 + i * 2 + 1
108

    
109
    return special
110

    
111

    
112
def _resolve_memory_number(number):
113
    if isinstance(number, str):
114
        return _get_special()[number]
115
    else:
116
        return number
117

    
118

    
119
def _wipe_memory(mem, char):
120
    mem.set_raw(char * (mem.size() / 8))
121

    
122

    
123
@directory.register
124
class IC2820Radio(icf.IcomCloneModeRadio, chirp_common.IcomDstarSupport):
125
    """Icom IC-2820"""
126
    VENDOR = "Icom"
127
    MODEL = "IC-2820H"
128

    
129
    _model = "\x29\x70\x00\x01"
130
    _memsize = 44224
131
    _endframe = "Icom Inc\x2e68"
132

    
133
    _ranges = [(0x0000, 0x6960, 32),
134
               (0x6960, 0x6980, 16),
135
               (0x6980, 0x7160, 32),
136
               (0x7160, 0x7180, 16),
137
               (0x7180, 0xACC0, 32),
138
               ]
139

    
140
    _num_banks = 26
141
    _bank_class = IC2820Bank
142
    _can_hispeed = True
143

    
144
    MYCALL_LIMIT = (1, 7)
145
    URCALL_LIMIT = (1, 61)
146
    RPTCALL_LIMIT = (1, 61)
147

    
148
    _memories = {}
149

    
150
    def _get_bank(self, loc):
151
        _bank = self._memobj.bank_info[loc]
152
        if _bank.bank == 0xFF:
153
            return None
154
        else:
155
            return _bank.bank
156

    
157
    def _set_bank(self, loc, bank):
158
        _bank = self._memobj.bank_info[loc]
159
        if bank is None:
160
            _bank.bank = 0xFF
161
        else:
162
            _bank.bank = bank
163

    
164
    def _get_bank_index(self, loc):
165
        _bank = self._memobj.bank_info[loc]
166
        return _bank.index
167

    
168
    def _set_bank_index(self, loc, index):
169
        _bank = self._memobj.bank_info[loc]
170
        _bank.index = index
171

    
172
    def get_features(self):
173
        rf = chirp_common.RadioFeatures()
174
        rf.has_settings = True
175
        rf.has_bank_index = True
176
        rf.has_bank_names = True
177
        rf.requires_call_lists = False
178
        rf.memory_bounds = (0, 499)
179
        rf.valid_modes = list(MODES)
180
        rf.valid_tmodes = list(TMODES)
181
        rf.valid_duplexes = list(set(DUPLEX))
182
        rf.valid_tuning_steps = list(chirp_common.TUNING_STEPS)
183
        rf.valid_bands = [(118000000, 999990000)]
184
        rf.valid_skips = ["", "S", "P"]
185
        rf.valid_characters = chirp_common.CHARSET_ASCII
186
        rf.valid_name_length = 8
187
        rf.valid_special_chans = sorted(_get_special().keys())
188

    
189
        return rf
190

    
191
    def process_mmap(self):
192
        self._memobj = bitwise.parse(MEM_FORMAT, self._mmap)
193

    
194
    def get_memory(self, number):
195
        number = _resolve_memory_number(number)
196

    
197
        bitpos = (1 << (number % 8))
198
        bytepos = number / 8
199

    
200
        _mem = self._memobj.memory[number]
201
        _used = self._memobj.used_flags[bytepos]
202

    
203
        is_used = ((_used & bitpos) == 0)
204

    
205
        if is_used and MODES[_mem.mode] == "DV":
206
            mem = chirp_common.DVMemory()
207
            mem.dv_urcall = str(_mem.urcall).rstrip()
208
            mem.dv_rpt1call = str(_mem.r1call).rstrip()
209
            mem.dv_rpt2call = str(_mem.r2call).rstrip()
210
        else:
211
            mem = chirp_common.Memory()
212

    
213
        mem.number = number
214
        if number < 500:
215
            _skip = self._memobj.skip_flags[bytepos]
216
            _pskip = self._memobj.pskip_flags[bytepos]
217
            if _skip & bitpos:
218
                mem.skip = "S"
219
            elif _pskip & bitpos:
220
                mem.skip = "P"
221
        else:
222
            mem.extd_number = util.get_dict_rev(_get_special(), number)
223
            mem.immutable = ["number", "skip", "bank", "bank_index",
224
                             "extd_number"]
225

    
226
        if not is_used:
227
            mem.empty = True
228
            return mem
229

    
230
        mem.freq = int(_mem.freq)
231
        mem.offset = int(_mem.offset)
232
        mem.rtone = chirp_common.TONES[_mem.rtone]
233
        mem.ctone = chirp_common.TONES[_mem.ctone]
234
        mem.tmode = TMODES[_mem.tmode]
235
        mem.duplex = DUPLEX[_mem.duplex]
236
        mem.mode = MODES[_mem.mode]
237
        mem.dtcs = chirp_common.DTCS_CODES[_mem.dtcs]
238
        mem.dtcs_polarity = DTCSP[_mem.dtcs_polarity]
239
        if _mem.tune_step > 8:
240
            mem.tuning_step = 5.0  # Sometimes TS is garbage?
241
        else:
242
            mem.tuning_step = chirp_common.TUNING_STEPS[_mem.tune_step]
243
        mem.name = str(_mem.name).rstrip()
244

    
245
        return mem
246

    
247
    def set_memory(self, mem):
248
        bitpos = (1 << (mem.number % 8))
249
        bytepos = mem.number / 8
250

    
251
        _mem = self._memobj.memory[mem.number]
252
        _used = self._memobj.used_flags[bytepos]
253

    
254
        was_empty = _used & bitpos
255

    
256
        if mem.number < 500:
257
            skip = self._memobj.skip_flags[bytepos]
258
            pskip = self._memobj.pskip_flags[bytepos]
259
            if mem.skip == "S":
260
                skip |= bitpos
261
            else:
262
                skip &= ~bitpos
263
            if mem.skip == "P":
264
                pskip |= bitpos
265
            else:
266
                pskip &= ~bitpos
267

    
268
        if mem.empty:
269
            _used |= bitpos
270
            _wipe_memory(_mem, "\xFF")
271
            self._set_bank(mem.number, None)
272
            return
273

    
274
        _used &= ~bitpos
275
        if was_empty:
276
            _wipe_memory(_mem, "\x00")
277

    
278
        _mem.freq = mem.freq
279
        _mem.offset = mem.offset
280
        _mem.rtone = chirp_common.TONES.index(mem.rtone)
281
        _mem.ctone = chirp_common.TONES.index(mem.ctone)
282
        _mem.tmode = TMODES.index(mem.tmode)
283
        _mem.duplex = DUPLEX.index(mem.duplex)
284
        _mem.mode = MODES.index(mem.mode)
285
        _mem.dtcs = chirp_common.DTCS_CODES.index(mem.dtcs)
286
        _mem.dtcs_polarity = DTCSP.index(mem.dtcs_polarity)
287
        _mem.tune_step = chirp_common.TUNING_STEPS.index(mem.tuning_step)
288
        _mem.name = mem.name.ljust(8)
289

    
290
        if isinstance(mem, chirp_common.DVMemory):
291
            _mem.urcall = mem.dv_urcall.ljust(8)
292
            _mem.r1call = mem.dv_rpt1call.ljust(8)
293
            _mem.r2call = mem.dv_rpt2call.ljust(8)
294

    
295
    def get_raw_memory(self, number):
296
        return repr(self._memobj.memory[number])
297

    
298
    def get_urcall_list(self):
299
        _calls = self._memobj.urcall
300
        calls = []
301

    
302
        for i in range(*self.URCALL_LIMIT):
303
            calls.append(str(_calls[i-1].call))
304

    
305
        return calls
306

    
307
    def get_repeater_call_list(self):
308
        _calls = self._memobj.rptcall
309
        calls = []
310

    
311
        for i in range(*self.RPTCALL_LIMIT):
312
            calls.append(str(_calls[i-1].call))
313

    
314
        return calls
315

    
316
    def get_mycall_list(self):
317
        _calls = self._memobj.mycall
318
        calls = []
319

    
320
        for i in range(*self.MYCALL_LIMIT):
321
            calls.append(str(_calls[i-1].call))
322

    
323
        return calls
324

    
325
    def set_urcall_list(self, calls):
326
        _calls = self._memobj.urcall
327

    
328
        for i in range(*self.URCALL_LIMIT):
329
            try:
330
                call = calls[i-1]
331
            except IndexError:
332
                call = " " * 8
333

    
334
            _calls[i-1].call = call.ljust(8)[:8]
335

    
336
    def set_repeater_call_list(self, calls):
337
        _calls = self._memobj.rptcall
338

    
339
        for i in range(*self.RPTCALL_LIMIT):
340
            try:
341
                call = calls[i-1]
342
            except IndexError:
343
                call = " " * 8
344

    
345
            _calls[i-1].call = call.ljust(8)[:8]
346

    
347
    def set_mycall_list(self, calls):
348
        _calls = self._memobj.mycall
349

    
350
        for i in range(*self.MYCALL_LIMIT):
351
            try:
352
                call = calls[i-1]
353
            except IndexError:
354
                call = " " * 8
355

    
356
            _calls[i-1].call = call.ljust(8)[:8]
(2-2/2)