Project

General

Profile

Bug #10274 » ft2900.py

Updated module - Dan Smith, 01/15/2023 03:48 AM

 
# Copyright 2011 Dan Smith <dsmith@danplanet.com>
#
# FT-2900-specific modifications by Richard Cochran, <ag6qr@sonic.net>
# Initial work on settings by Chris Fosnight, <chris.fosnight@gmail.com>
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.

import time
import os
import logging

from chirp import util, memmap, chirp_common, bitwise, directory, errors
from chirp.drivers.yaesu_clone import YaesuCloneModeRadio
from chirp.settings import RadioSetting, RadioSettingGroup, \
RadioSettingValueList, RadioSettingValueString, RadioSettings

from textwrap import dedent

LOG = logging.getLogger(__name__)


def _send(s, data):
s.write(data)
echo = s.read(len(data))
if data != echo:
raise Exception("Failed to read echo")
LOG.debug("got echo\n%s\n" % util.hexprint(echo))

ACK = b"\x06"
INITIAL_CHECKSUM = 0


def _download(radio):

blankChunk = b""
for _i in range(0, 32):
blankChunk += b"\xff"

LOG.debug("in _download\n")

data = b""
for _i in range(0, 20):
data = radio.pipe.read(20)
LOG.debug("Header:\n%s" % util.hexprint(data))
LOG.debug("len(header) = %s\n" % len(data))

if data == radio.IDBLOCK:
break

if data != radio.IDBLOCK:
raise Exception("Failed to read header")

_send(radio.pipe, ACK)

# initialize data, the big var that holds all memory
data = b""

_blockNum = 0

while len(data) < radio._block_sizes[1]:
_blockNum += 1
time.sleep(0.03)
chunk = radio.pipe.read(32)
LOG.debug("Block %i " % (_blockNum))
if chunk == blankChunk:
LOG.debug("blank chunk\n")
else:
LOG.debug("Got: %i:\n%s" % (len(chunk), util.hexprint(chunk)))
if len(chunk) != 32:
LOG.debug("len chunk is %i\n" % (len(chunk)))
raise Exception("Failed to get full data block")
break
else:
data += chunk

if radio.status_fn:
status = chirp_common.Status()
status.max = radio._block_sizes[1]
status.cur = len(data)
status.msg = "Cloning from radio"
radio.status_fn(status)

LOG.debug("Total: %i" % len(data))

# radio should send us one final termination byte, containing
# checksum
chunk = radio.pipe.read(32)
if len(chunk) != 1:
LOG.debug("len(chunk) is %i\n" % len(chunk))
raise Exception("radio sent extra unknown data")
LOG.debug("Got: %i:\n%s" % (len(chunk), util.hexprint(chunk)))

# compute checksum
cs = INITIAL_CHECKSUM
for byte in radio.IDBLOCK:
cs += byte
for byte in data:
cs += byte
LOG.debug("calculated checksum is %x\n" % (cs & 0xff))
LOG.debug("Radio sent checksum is %x\n" % chunk[0])

if (cs & 0xff) != chunk[0]:
raise Exception("Failed checksum on read.")

# for debugging purposes, dump the channels, in hex.
for _i in range(0, 200):
_startData = 1892 + 20 * _i
chunk = data[_startData:_startData + 20]
LOG.debug("channel %i:\n%s" % (_i, util.hexprint(chunk)))

return memmap.MemoryMapBytes(data)


def _upload(radio):
for _i in range(0, 10):
data = radio.pipe.read(256)
if not data:
break
LOG.debug("What is this garbage?\n%s" % util.hexprint(data))
raise Exception("Radio sent unrecognized data")

_send(radio.pipe, radio.IDBLOCK)
time.sleep(.2)
ack = radio.pipe.read(300)
LOG.debug("Ack was (%i):\n%s" % (len(ack), util.hexprint(ack)))
if ack != ACK:
raise Exception("Radio did not ack ID. Check cable, verify"
" radio is not locked.\n"
" (press & Hold red \"*L\" button to unlock"
" radio if needed)")

block = 0
cs = INITIAL_CHECKSUM
for byte in radio.IDBLOCK:
cs += byte

while block < (radio.get_memsize() // 32):
data = radio.get_mmap()[block * 32:(block + 1) * 32]

LOG.debug("Writing block %i:\n%s" % (block, util.hexprint(data)))

_send(radio.pipe, data)
time.sleep(0.03)

for byte in data:
cs += byte

if radio.status_fn:
status = chirp_common.Status()
status.max = radio._block_sizes[1]
status.cur = block * 32
status.msg = "Cloning to radio"
radio.status_fn(status)
block += 1

_send(radio.pipe, bytes([cs & 0xFF]))

MEM_FORMAT = """
#seekto 0x0080;
struct {
u8 apo;
u8 arts_beep;
u8 bell;
u8 dimmer;
u8 cw_id_string[16];
u8 cw_trng;
u8 x95;
u8 x96;
u8 x97;
u8 int_cd;
u8 int_set;
u8 x9A;
u8 x9B;
u8 lock;
u8 x9D;
u8 mic_gain;
u8 open_msg;
u8 openMsg_Text[6];
u8 rf_sql;
u8 unk:6,
pag_abk:1,
unk:1;
u8 pag_cdr_1;
u8 pag_cdr_2;
u8 pag_cdt_1;
u8 pag_cdt_2;
u8 prog_p1;
u8 xAD;
u8 prog_p2;
u8 xAF;
u8 prog_p3;
u8 xB1;
u8 prog_p4;
u8 xB3;
u8 resume;
u8 tot;
u8 unk:1,
cw_id:1,
unk:1,
ts_speed:1,
ars:1,
unk:2,
dtmf_mode:1;
u8 unk:1,
ts_mut:1
wires_auto:1,
busy_lockout:1,
edge_beep:1,
unk:3;
u8 unk:2,
s_search:1,
unk:2,
cw_trng_units:1,
unk:2;
u8 dtmf_speed:1,
unk:2,
arts_interval:1,
unk:1,
inverted_dcs:1,
unk:1,
mw_mode:1;
u8 unk:2,
wires_mode:1,
wx_alert:1,
unk:1,
wx_vol_max:1,
revert:1,
unk:1;
u8 vfo_scan;
u8 scan_mode;
u8 dtmf_delay;
u8 beep;
u8 xBF;
} settings;

#seekto 0x00d0;
u8 passwd[4];
u8 mbs;

#seekto 0x00c0;
struct {
u16 in_use;
} bank_used[8];

#seekto 0x00ef;
u8 currentTone;

#seekto 0x00f0;
u8 curChannelMem[20];

#seekto 0x1e0;
struct {
u8 dtmf_string[16];
} dtmf_strings[10];

#seekto 0x0127;
u8 curChannelNum;

#seekto 0x012a;
u8 banksoff1;

#seekto 0x15f;
u8 checksum1;

#seekto 0x16f;
u8 curentTone2;

#seekto 0x1aa;
u16 banksoff2;

#seekto 0x1df;
u8 checksum2;

#seekto 0x0360;
struct{
u8 name[6];
} bank_names[8];


#seekto 0x03c4;
struct{
u16 channels[50];
} banks[8];

#seekto 0x06e4;
struct {
u8 even_pskip:1,
even_skip:1,
even_valid:1,
even_masked:1,
odd_pskip:1,
odd_skip:1,
odd_valid:1,
odd_masked:1;
} flags[225];

#seekto 0x0764;
struct {
u8 unknown0:2,
isnarrow:1,
unknown1:5;
u8 unknown2:2,
duplex:2,
unknown3:1,
step:3;
bbcd freq[3];
u8 power:2,
unknown4:3,
tmode:3;
u8 name[6];
bbcd offset[3];
u8 ctonesplitflag:1,
ctone:7;
u8 rx_dtcssplitflag:1,
rx_dtcs:7;
u8 unknown5;
u8 rtonesplitflag:1,
rtone:7;
u8 dtcssplitflag:1,
dtcs:7;
} memory[200];

"""

MODES = ["FM", "NFM"]
TMODES = ["", "Tone", "TSQL", "DTCS", "TSQL-R", "Cross"]
CROSS_MODES = ["DTCS->", "Tone->DTCS", "DTCS->Tone",
"Tone->Tone", "DTCS->DTCS"]
DUPLEX = ["", "-", "+", "split"]
POWER_LEVELS = [chirp_common.PowerLevel("Hi", watts=75),
chirp_common.PowerLevel("Low3", watts=30),
chirp_common.PowerLevel("Low2", watts=10),
chirp_common.PowerLevel("Low1", watts=5),
]

CHARSET = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ +-/?C[] _"
STEPS = [5.0, 10.0, 12.5, 15.0, 20.0, 25.0, 50.0, 100.0]


def _decode_tone(radiotone):
try:
chirptone = chirp_common.TONES[radiotone]
except IndexError:
chirptone = 100
LOG.debug("found invalid radio tone: %i\n" % radiotone)
return chirptone


def _decode_dtcs(radiodtcs):
try:
chirpdtcs = chirp_common.DTCS_CODES[radiodtcs]
except IndexError:
chirpdtcs = 23
LOG.debug("found invalid radio dtcs code: %i\n" % radiodtcs)
return chirpdtcs


def _decode_name(mem):
name = ""
for i in mem:
if (i & 0x7F) == 0x7F:
break
try:
name += CHARSET[i & 0x7F]
except IndexError:
LOG.debug("Unknown char index: %x " % (i))
name = name.strip()
return name


def _encode_name(mem):
if(mem.strip() == ""):
return [0xff] * 6

name = [None] * 6
for i in range(0, 6):
try:
name[i] = CHARSET.index(mem[i])
except IndexError:
name[i] = CHARSET.index(" ")

name[0] = name[0] | 0x80
return name


def _wipe_memory(mem):
mem.set_raw(b"\xff" * (mem.size() // 8))


class FT2900Bank(chirp_common.NamedBank):

def get_name(self):
_bank = self._model._radio._memobj.bank_names[self.index]
name = ""
for i in _bank.name:
if i == 0xff:
break
name += CHARSET[i & 0x7f]

return name.rstrip()

def set_name(self, name):
name = name.upper().ljust(6)[:6]
_bank = self._model._radio._memobj.bank_names[self.index]
_bank.name = [CHARSET.index(x) for x in name.ljust(6)[:6]]


class FT2900BankModel(chirp_common.BankModel):

def get_num_mappings(self):
return 8

def get_mappings(self):
banks = self._radio._memobj.banks
bank_mappings = []
for index, _bank in enumerate(banks):
bank = FT2900Bank(self, "%i" % index, "b%i" % (index + 1))
bank.index = index
bank_mappings.append(bank)

return bank_mappings

def _get_channel_numbers_in_bank(self, bank):
_bank_used = self._radio._memobj.bank_used[bank.index]
if _bank_used.in_use == 0xffff:
return set()

_members = self._radio._memobj.banks[bank.index]
return set([int(ch) for ch in _members.channels if ch != 0xffff])

def _update_bank_with_channel_numbers(self, bank, channels_in_bank):
_members = self._radio._memobj.banks[bank.index]
if len(channels_in_bank) > len(_members.channels):
raise Exception("More than %i entries in bank %d" %
(len(_members.channels), bank.index))

empty = 0
for index, channel_number in enumerate(sorted(channels_in_bank)):
_members.channels[index] = channel_number
empty = index + 1
for index in range(empty, len(_members.channels)):
_members.channels[index] = 0xffff

_bank_used = self._radio._memobj.bank_used[bank.index]
if empty == 0:
_bank_used.in_use = 0xffff
else:
_bank_used.in_use = empty - 1

def add_memory_to_mapping(self, memory, bank):
channels_in_bank = self._get_channel_numbers_in_bank(bank)
channels_in_bank.add(memory.number)
self._update_bank_with_channel_numbers(bank, channels_in_bank)

# tells radio that banks are active
self._radio._memobj.banksoff1 = bank.index
self._radio._memobj.banksoff2 = bank.index

def remove_memory_from_mapping(self, memory, bank):
channels_in_bank = self._get_channel_numbers_in_bank(bank)
try:
channels_in_bank.remove(memory.number)
except KeyError:
raise Exception("Memory %i is not in bank %s. Cannot remove" %
(memory.number, bank))
self._update_bank_with_channel_numbers(bank, channels_in_bank)

def get_mapping_memories(self, bank):
memories = []
for channel in self._get_channel_numbers_in_bank(bank):
memories.append(self._radio.get_memory(channel))

return memories

def get_memory_mappings(self, memory):
banks = []
for bank in self.get_mappings():
if memory.number in self._get_channel_numbers_in_bank(bank):
banks.append(bank)

return banks


@directory.register
class FT2900Radio(YaesuCloneModeRadio):

"""Yaesu FT-2900"""
VENDOR = "Yaesu"
MODEL = "FT-2900R/1900R"
IDBLOCK = b"\x56\x43\x32\x33\x00\x02\x46\x01\x01\x01"
BAUD_RATE = 19200

_memsize = 8000
_block_sizes = [8, 8000]

def get_features(self):
rf = chirp_common.RadioFeatures()

rf.memory_bounds = (0, 199)

rf.can_odd_split = True
rf.has_ctone = True
rf.has_rx_dtcs = True
rf.has_cross = True
rf.has_dtcs_polarity = False
rf.has_bank = True
rf.has_bank_names = True
rf.has_settings = True

rf.valid_tuning_steps = STEPS
rf.valid_modes = MODES
rf.valid_tmodes = TMODES
rf.valid_cross_modes = CROSS_MODES
rf.valid_bands = [(136000000, 174000000)]
rf.valid_power_levels = POWER_LEVELS
rf.valid_duplexes = DUPLEX
rf.valid_skips = ["", "S", "P"]
rf.valid_name_length = 6
rf.valid_characters = CHARSET

return rf

def sync_in(self):
start = time.time()
try:
self._mmap = _download(self)
except errors.RadioError:
raise
except Exception as e:
raise errors.RadioError("Failed to communicate with radio: %s" % e)
LOG.info("Downloaded in %.2f sec" % (time.time() - start))
self.process_mmap()

def sync_out(self):
self.pipe.timeout = 1
start = time.time()
try:
_upload(self)
except errors.RadioError:
raise
except Exception as e:
raise errors.RadioError("Failed to communicate with radio: %s" % e)
LOG.info("Uploaded in %.2f sec" % (time.time() - start))

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

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

def get_memory(self, number):
_mem = self._memobj.memory[number]
_flag = self._memobj.flags[(number) / 2]

nibble = ((number) % 2) and "even" or "odd"
used = _flag["%s_masked" % nibble]
valid = _flag["%s_valid" % nibble]
pskip = _flag["%s_pskip" % nibble]
skip = _flag["%s_skip" % nibble]

mem = chirp_common.Memory()

mem.number = number

if _mem.get_raw()[0] == "\xFF" or not valid or not used:
mem.empty = True
return mem

mem.tuning_step = STEPS[_mem.step]
mem.freq = int(_mem.freq) * 1000

# compensate for 12.5 kHz tuning steps, add 500 Hz if needed
if(mem.tuning_step == 12.5):
lastdigit = int(_mem.freq) % 10
if (lastdigit == 2 or lastdigit == 7):
mem.freq += 500

mem.offset = chirp_common.fix_rounded_step(int(_mem.offset) * 1000)
mem.duplex = DUPLEX[_mem.duplex]
if _mem.tmode < TMODES.index("Cross"):
mem.tmode = TMODES[_mem.tmode]
mem.cross_mode = CROSS_MODES[0]
else:
mem.tmode = "Cross"
mem.cross_mode = CROSS_MODES[_mem.tmode - TMODES.index("Cross")]

mem.rtone = _decode_tone(_mem.rtone)
mem.ctone = _decode_tone(_mem.ctone)

# check for unequal ctone/rtone in TSQL mode. map it as a
# cross tone mode
if mem.rtone != mem.ctone and (mem.tmode == "TSQL" or
mem.tmode == "Tone"):
mem.tmode = "Cross"
mem.cross_mode = "Tone->Tone"

mem.dtcs = _decode_dtcs(_mem.dtcs)
mem.rx_dtcs = _decode_dtcs(_mem.rx_dtcs)

# check for unequal dtcs/rx_dtcs in DTCS mode. map it as a
# cross tone mode
if mem.dtcs != mem.rx_dtcs and mem.tmode == "DTCS":
mem.tmode = "Cross"
mem.cross_mode = "DTCS->DTCS"

if (int(_mem.name[0]) & 0x80) != 0:
mem.name = _decode_name(_mem.name)

mem.mode = _mem.isnarrow and "NFM" or "FM"
mem.skip = pskip and "P" or skip and "S" or ""
mem.power = POWER_LEVELS[3 - _mem.power]

return mem

def set_memory(self, mem):
_mem = self._memobj.memory[mem.number]
_flag = self._memobj.flags[(mem.number) / 2]

nibble = ((mem.number) % 2) and "even" or "odd"

valid = _flag["%s_valid" % nibble]
used = _flag["%s_masked" % nibble]

if not valid:
_wipe_memory(_mem)

if mem.empty and valid and not used:
_flag["%s_valid" % nibble] = False
return

_flag["%s_masked" % nibble] = not mem.empty

if mem.empty:
return

_flag["%s_valid" % nibble] = True

_mem.freq = mem.freq / 1000
_mem.offset = mem.offset / 1000
_mem.duplex = DUPLEX.index(mem.duplex)

# clear all the split tone flags -- we'll set them as needed below
_mem.ctonesplitflag = 0
_mem.rx_dtcssplitflag = 0
_mem.rtonesplitflag = 0
_mem.dtcssplitflag = 0

if mem.tmode != "Cross":
_mem.tmode = TMODES.index(mem.tmode)
# for the non-cross modes, use ONE tone for both send
# and receive but figure out where to get it from.
if mem.tmode == "TSQL" or mem.tmode == "TSQL-R":
_mem.rtone = chirp_common.TONES.index(mem.ctone)
_mem.ctone = chirp_common.TONES.index(mem.ctone)
else:
_mem.rtone = chirp_common.TONES.index(mem.rtone)
_mem.ctone = chirp_common.TONES.index(mem.rtone)

# and one tone for dtcs, but this is always the sending one
_mem.dtcs = chirp_common.DTCS_CODES.index(mem.dtcs)
_mem.rx_dtcs = chirp_common.DTCS_CODES.index(mem.dtcs)

else:
_mem.rtone = chirp_common.TONES.index(mem.rtone)
_mem.ctone = chirp_common.TONES.index(mem.ctone)
_mem.dtcs = chirp_common.DTCS_CODES.index(mem.dtcs)
_mem.rx_dtcs = chirp_common.DTCS_CODES.index(mem.rx_dtcs)
if mem.cross_mode == "Tone->Tone":
# tone->tone cross mode is treated as
# TSQL, but with separate tones for
# send and receive
_mem.tmode = TMODES.index("TSQL")
_mem.rtonesplitflag = 1
elif mem.cross_mode == "DTCS->DTCS":
# DTCS->DTCS cross mode is treated as
# DTCS, but with separate codes for
# send and receive
_mem.tmode = TMODES.index("DTCS")
_mem.dtcssplitflag = 1
else:
_mem.tmode = TMODES.index("Cross") + \
CROSS_MODES.index(mem.cross_mode)

_mem.isnarrow = MODES.index(mem.mode)
_mem.step = STEPS.index(mem.tuning_step)
_flag["%s_pskip" % nibble] = mem.skip == "P"
_flag["%s_skip" % nibble] = mem.skip == "S"
if mem.power:
_mem.power = 3 - POWER_LEVELS.index(mem.power)
else:
_mem.power = 3

_mem.name = _encode_name(mem.name)

# set all unknown areas of the memory map to 0
_mem.unknown0 = 0
_mem.unknown1 = 0
_mem.unknown2 = 0
_mem.unknown3 = 0
_mem.unknown4 = 0
_mem.unknown5 = 0

LOG.debug("encoded mem\n%s\n" % (util.hexprint(_mem.get_raw()[0:20])))

def get_settings(self):
_settings = self._memobj.settings
_dtmf_strings = self._memobj.dtmf_strings
_passwd = self._memobj.passwd

repeater = RadioSettingGroup("repeater", "Repeater Settings")
ctcss = RadioSettingGroup("ctcss", "CTCSS/DCS/EPCS Settings")
arts = RadioSettingGroup("arts", "ARTS Settings")
mbls = RadioSettingGroup("banks", "Memory Settings")
scan = RadioSettingGroup("scan", "Scan Settings")
dtmf = RadioSettingGroup("dtmf", "DTMF Settings")
wires = RadioSettingGroup("wires", "WiRES(tm) Settings")
switch = RadioSettingGroup("switch", "Switch/Knob Settings")
disp = RadioSettingGroup("disp", "Display Settings")
misc = RadioSettingGroup("misc", "Miscellaneous Settings")

setmode = RadioSettings(repeater, ctcss, arts, mbls, scan,
dtmf, wires, switch, disp, misc)

# numbers and names of settings refer to the way they're
# presented in the set menu, as well as the list starting on
# page 74 of the manual

# 1 APO
opts = ["Off", "30 Min", "1 Hour", "3 Hour", "5 Hour", "8 Hour"]
misc.append(
RadioSetting(
"apo", "Automatic Power Off",
RadioSettingValueList(opts, opts[_settings.apo])))

# 2 AR.BEP
opts = ["Off", "In Range", "Always"]
arts.append(
RadioSetting(
"arts_beep", "ARTS Beep",
RadioSettingValueList(opts, opts[_settings.arts_beep])))

# 3 AR.INT
opts = ["15 Sec", "25 Sec"]
arts.append(
RadioSetting(
"arts_interval", "ARTS Polling Interval",
RadioSettingValueList(opts, opts[_settings.arts_interval])))

# 4 ARS
opts = ["Off", "On"]
repeater.append(
RadioSetting(
"ars", "Automatic Repeater Shift",
RadioSettingValueList(opts, opts[_settings.ars])))

# 5 BCLO
opts = ["Off", "On"]
misc.append(RadioSetting(
"busy_lockout", "Busy Channel Lock-Out",
RadioSettingValueList(opts, opts[_settings.busy_lockout])))

# 6 BEEP
opts = ["Off", "Key+Scan", "Key"]
switch.append(RadioSetting(
"beep", "Enable the Beeper",
RadioSettingValueList(opts, opts[_settings.beep])))

# 7 BELL
opts = ["Off", "1", "3", "5", "8", "Continuous"]
ctcss.append(RadioSetting("bell", "Bell Repetitions",
RadioSettingValueList(opts, opts[
_settings.bell])))

# 8 BNK.LNK
for i in range(0, 8):
opts = ["Off", "On"]
mbs = (self._memobj.mbs >> i) & 1
rs = RadioSetting("mbs%i" % i, "Bank %s Scan" % (i + 1),
RadioSettingValueList(opts, opts[mbs]))

def apply_mbs(s, index):
if int(s.value):
self._memobj.mbs |= (1 << index)
else:
self._memobj.mbs &= ~(1 << index)
rs.set_apply_callback(apply_mbs, i)
mbls.append(rs)

# 9 BNK.NM - A per-bank attribute, nothing to do here.

# 10 CLK.SFT - A per-channel attribute, nothing to do here.

# 11 CW.ID
opts = ["Off", "On"]
arts.append(RadioSetting("cw_id", "CW ID Enable",
RadioSettingValueList(opts, opts[
_settings.cw_id])))

cw_id_text = ""
for i in _settings.cw_id_string:
try:
cw_id_text += CHARSET[i & 0x7F]
except IndexError:
if i != 0xff:
LOG.debug("unknown char index in cw id: %x " % (i))

val = RadioSettingValueString(0, 16, cw_id_text, True)
val.set_charset(CHARSET + "abcdefghijklmnopqrstuvwxyz")
rs = RadioSetting("cw_id_string", "CW Identifier Text", val)

def apply_cw_id(s):
str = s.value.get_value().upper().rstrip()
mval = ""
mval = [chr(CHARSET.index(x)) for x in str]
for x in range(len(mval), 16):
mval.append(chr(0xff))
for x in range(0, 16):
_settings.cw_id_string[x] = ord(mval[x])
rs.set_apply_callback(apply_cw_id)
arts.append(rs)

# 12 CWTRNG
opts = ["Off", "4WPM", "5WPM", "6WPM", "7WPM", "8WPM", "9WPM",
"10WPM", "11WPM", "12WPM", "13WPM", "15WPM", "17WPM",
"20WPM", "24WPM", "30WPM", "40WPM"]
misc.append(RadioSetting("cw_trng", "CW Training",
RadioSettingValueList(opts, opts[
_settings.cw_trng])))

# todo: make the setting of the units here affect the display
# of the speed. Not critical, but would be slick.
opts = ["CPM", "WPM"]
misc.append(RadioSetting("cw_trng_units", "CW Training Units",
RadioSettingValueList(opts,
opts[_settings.
cw_trng_units])))

# 13 DC VLT - a read-only status, so nothing to do here

# 14 DCS CD - A per-channel attribute, nothing to do here

# 15 DCS.RV
opts = ["Disabled", "Enabled"]
ctcss.append(RadioSetting(
"inverted_dcs",
"\"Inverted\" DCS Code Decoding",
RadioSettingValueList(opts,
opts[_settings.inverted_dcs])))

# 16 DIMMER
opts = ["Off"] + ["Level %d" % (x) for x in range(1, 11)]
disp.append(RadioSetting("dimmer", "Dimmer",
RadioSettingValueList(opts,
opts[_settings
.dimmer])))

# 17 DT.A/M
opts = ["Manual", "Auto"]
dtmf.append(RadioSetting("dtmf_mode", "DTMF Autodialer",
RadioSettingValueList(opts,
opts[_settings
.dtmf_mode])))

# 18 DT.DLY
opts = ["50 ms", "250 ms", "450 ms", "750 ms", "1000 ms"]
dtmf.append(RadioSetting("dtmf_delay", "DTMF Autodialer Delay Time",
RadioSettingValueList(opts,
opts[_settings
.dtmf_delay])))

# 19 DT.SET
for memslot in range(0, 10):
dtmf_memory = ""
for i in _dtmf_strings[memslot].dtmf_string:
if i != 0xFF:
try:
dtmf_memory += CHARSET[i]
except IndexError:
LOG.debug("unknown char index in dtmf: %x " % (i))

val = RadioSettingValueString(0, 16, dtmf_memory, True)
val.set_charset(CHARSET + "abcdef")
rs = RadioSetting("dtmf_string_%d" % memslot,
"DTMF Memory %d" % memslot, val)

def apply_dtmf(s, i):
LOG.debug("applying dtmf for %x\n" % i)
str = s.value.get_value().upper().rstrip()
LOG.debug("str is %s\n" % str)
mval = ""
mval = [chr(CHARSET.index(x)) for x in str]
for x in range(len(mval), 16):
mval.append(chr(0xff))
for x in range(0, 16):
_dtmf_strings[i].dtmf_string[x] = ord(mval[x])
rs.set_apply_callback(apply_dtmf, memslot)
dtmf.append(rs)

# 20 DT.SPD
opts = ["50 ms", "100 ms"]
dtmf.append(RadioSetting("dtmf_speed",
"DTMF Autodialer Sending Speed",
RadioSettingValueList(opts,
opts[_settings.
dtmf_speed])))

# 21 EDG.BEP
opts = ["Off", "On"]
mbls.append(RadioSetting("edge_beep", "Band Edge Beeper",
RadioSettingValueList(opts,
opts[_settings.
edge_beep])))

# 22 INT.CD
opts = ["DTMF %X" % (x) for x in range(0, 16)]
wires.append(RadioSetting("int_cd", "Access Number for WiRES(TM)",
RadioSettingValueList(opts, opts[
_settings.int_cd])))

# 23 ING MD
opts = ["Sister Radio Group", "Friends Radio Group"]
wires.append(RadioSetting("wires_mode",
"Internet Link Connection Mode",
RadioSettingValueList(opts,
opts[_settings.
wires_mode])))

# 24 INT.A/M
opts = ["Manual", "Auto"]
wires.append(RadioSetting("wires_auto", "Internet Link Autodialer",
RadioSettingValueList(opts,
opts[_settings
.wires_auto])))
# 25 INT.SET
opts = ["F%d" % (x) for x in range(0, 10)]

wires.append(RadioSetting("int_set", "Memory Register for "
"non-WiRES Internet",
RadioSettingValueList(opts,
opts[_settings
.int_set])))

# 26 LOCK
opts = ["Key", "Dial", "Key + Dial", "PTT",
"Key + PTT", "Dial + PTT", "All"]
switch.append(RadioSetting("lock", "Control Locking",
RadioSettingValueList(opts,
opts[_settings
.lock])))

# 27 MCGAIN
opts = ["Level %d" % (x) for x in range(1, 10)]
misc.append(RadioSetting("mic_gain", "Microphone Gain",
RadioSettingValueList(opts,
opts[_settings
.mic_gain])))

# 28 MEM.SCN
opts = ["Tag 1", "Tag 2", "All Channels"]
rs = RadioSetting("scan_mode", "Memory Scan Mode",
RadioSettingValueList(opts,
opts[_settings
.scan_mode - 1]))
# this setting is unusual in that it starts at 1 instead of 0.
# that is, index 1 corresponds to "Tag 1", and index 0 is invalid.
# so we create a custom callback to handle this.

def apply_scan_mode(s):
myopts = ["Tag 1", "Tag 2", "All Channels"]
_settings.scan_mode = myopts.index(s.value.get_value()) + 1
rs.set_apply_callback(apply_scan_mode)
mbls.append(rs)

# 29 MW MD
opts = ["Lower", "Next"]
mbls.append(RadioSetting("mw_mode", "Memory Write Mode",
RadioSettingValueList(opts,
opts[_settings
.mw_mode])))

# 30 NM SET - This is per channel, so nothing to do here

# 31 OPN.MSG
opts = ["Off", "DC Supply Voltage", "Text Message"]
disp.append(RadioSetting("open_msg", "Opening Message Type",
RadioSettingValueList(opts,
opts[_settings.
open_msg])))

openmsg = ""
for i in _settings.openMsg_Text:
try:
openmsg += CHARSET[i & 0x7F]
except IndexError:
if i != 0xff:
LOG.debug("unknown char index in openmsg: %x " % (i))

val = RadioSettingValueString(0, 6, openmsg, True)
val.set_charset(CHARSET + "abcdefghijklmnopqrstuvwxyz")
rs = RadioSetting("openMsg_Text", "Opening Message Text", val)

def apply_openmsg(s):
str = s.value.get_value().upper().rstrip()
mval = ""
mval = [chr(CHARSET.index(x)) for x in str]
for x in range(len(mval), 6):
mval.append(chr(0xff))
for x in range(0, 6):
_settings.openMsg_Text[x] = ord(mval[x])
rs.set_apply_callback(apply_openmsg)
disp.append(rs)

# 32 PAGER - a per-channel attribute

# 33 PAG.ABK
opts = ["Off", "On"]
ctcss.append(RadioSetting("pag_abk", "Paging Answer Back",
RadioSettingValueList(opts,
opts[_settings
.pag_abk])))

# 34 PAG.CDR
opts = ["%2.2d" % (x) for x in range(1, 50)]
ctcss.append(RadioSetting("pag_cdr_1", "Receive Page Code 1",
RadioSettingValueList(opts,
opts[_settings
.pag_cdr_1])))

ctcss.append(RadioSetting("pag_cdr_2", "Receive Page Code 2",
RadioSettingValueList(opts,
opts[_settings
.pag_cdr_2])))

# 35 PAG.CDT
opts = ["%2.2d" % (x) for x in range(1, 50)]
ctcss.append(RadioSetting("pag_cdt_1", "Transmit Page Code 1",
RadioSettingValueList(opts,
opts[_settings
.pag_cdt_1])))

ctcss.append(RadioSetting("pag_cdt_2", "Transmit Page Code 2",
RadioSettingValueList(opts,
opts[_settings
.pag_cdt_2])))

# Common Button Options
button_opts = ["Squelch Off", "Weather", "Smart Search",
"Tone Scan", "Scan", "T Call", "ARTS"]

# 36 PRG P1
opts = button_opts + ["DC Volts"]
switch.append(RadioSetting(
"prog_p1", "P1 Button",
RadioSettingValueList(opts, opts[_settings.prog_p1])))

# 37 PRG P2
opts = button_opts + ["Dimmer"]
switch.append(RadioSetting(
"prog_p2", "P2 Button",
RadioSettingValueList(opts, opts[_settings.prog_p2])))

# 38 PRG P3
opts = button_opts + ["Mic Gain"]
switch.append(RadioSetting(
"prog_p3", "P3 Button",
RadioSettingValueList(opts, opts[_settings.prog_p3])))

# 39 PRG P4
opts = button_opts + ["Skip"]
switch.append(RadioSetting(
"prog_p4", "P4 Button",
RadioSettingValueList(opts, opts[_settings.prog_p4])))

# 40 PSWD
password = ""
for i in _passwd:
if i != 0xFF:
try:
password += CHARSET[i]
except IndexError:
LOG.debug("unknown char index in password: %x " % (i))

val = RadioSettingValueString(0, 4, password, True)
val.set_charset(CHARSET[0:15] + "abcdef ")
rs = RadioSetting("passwd", "Password", val)

def apply_password(s):
str = s.value.get_value().upper().rstrip()
mval = ""
mval = [chr(CHARSET.index(x)) for x in str]
for x in range(len(mval), 4):
mval.append(chr(0xff))
for x in range(0, 4):
_passwd[x] = ord(mval[x])
rs.set_apply_callback(apply_password)
misc.append(rs)

# 41 RESUME
opts = ["3 Sec", "5 Sec", "10 Sec", "Busy", "Hold"]
scan.append(RadioSetting("resume", "Scan Resume Mode",
RadioSettingValueList(opts, opts[
_settings.resume])))

# 42 RF.SQL
opts = ["Off"] + ["S-%d" % (x) for x in range(1, 10)]
misc.append(RadioSetting("rf_sql", "RF Squelch Threshold",
RadioSettingValueList(opts, opts[
_settings.rf_sql])))

# 43 RPT - per channel attribute, nothing to do here

# 44 RVRT
opts = ["Off", "On"]
misc.append(RadioSetting("revert", "Priority Revert",
RadioSettingValueList(opts, opts[
_settings.revert])))

# 45 S.SRCH
opts = ["Single", "Continuous"]
misc.append(RadioSetting("s_search", "Smart Search Sweep Mode",
RadioSettingValueList(opts, opts[
_settings.s_search])))

# 46 SHIFT - per channel setting, nothing to do here

# 47 SKIP = per channel setting, nothing to do here

# 48 SPLIT - per channel attribute, nothing to do here

# 49 SQL.TYP - per channel attribute, nothing to do here

# 50 STEP - per channel attribute, nothing to do here

# 51 TEMP - read-only status, nothing to do here

# 52 TN FRQ - per channel attribute, nothing to do here

# 53 TOT
opts = ["Off", "1 Min", "3 Min", "5 Min", "10 Min"]
misc.append(RadioSetting("tot", "Timeout Timer",
RadioSettingValueList(opts,
opts[_settings.tot])))

# 54 TS MUT
opts = ["Off", "On"]
ctcss.append(RadioSetting("ts_mut", "Tone Search Mute",
RadioSettingValueList(opts,
opts[_settings
.ts_mut])))

# 55 TS SPEED
opts = ["Fast", "Slow"]
ctcss.append(RadioSetting("ts_speed", "Tone Search Scanner Speed",
RadioSettingValueList(opts,
opts[_settings
.ts_speed])))

# 56 VFO.SCN
opts = ["+/- 1MHz", "+/- 2MHz", "+/-5MHz", "All"]
scan.append(RadioSetting("vfo_scan", "VFO Scanner Width",
RadioSettingValueList(opts,
opts[_settings
.vfo_scan])))

# 57 WX.ALT
opts = ["Off", "On"]
misc.append(RadioSetting("wx_alert", "Weather Alert Scan",
RadioSettingValueList(opts, opts[
_settings.wx_alert])))

# 58 WX.VOL
opts = ["Normal", "Maximum"]
misc.append(RadioSetting("wx_vol_max", "Weather Alert Volume",
RadioSettingValueList(opts, opts[
_settings.wx_vol_max])))

# 59 W/N DV - this is a per-channel attribute, nothing to do here

return setmode

def set_settings(self, uisettings):
_settings = self._memobj.settings
for element in uisettings:
if not isinstance(element, RadioSetting):
self.set_settings(element)
continue
if not element.changed():
continue

try:
name = element.get_name()
value = element.value

if element.has_apply_callback():
LOG.debug("Using apply callback")
element.run_apply_callback()
else:
obj = getattr(_settings, name)
setattr(_settings, name, value)

LOG.debug("Setting %s: %s" % (name, value))
except Exception as e:
LOG.debug(element.get_name())
raise

def get_bank_model(self):
return FT2900BankModel(self)

@classmethod
def match_model(cls, filedata, filename):
return len(filedata) == cls._memsize

@classmethod
def get_prompts(cls):
rp = chirp_common.RadioPrompts()
rp.pre_download = _(dedent("""\
1. Turn Radio off.
2. Connect data cable.
3. While holding "A/N LOW" button, turn radio on.
4. <b>After clicking OK</b>, press "SET MHz" to send image."""))
rp.pre_upload = _(dedent("""\
1. Turn Radio off.
2. Connect data cable.
3. While holding "A/N LOW" button, turn radio on.
4. Press "MW D/MR" to receive image.
5. Make sure display says "-WAIT-" (see note below if not)
6. Click OK to dismiss this dialog and start transfer.

Note: if you don't see "-WAIT-" at step 5, try cycling
power and pressing and holding red "*L" button to unlock
radio, then start back at step 1."""))
return rp


# the FT2900E is the European version of the radio, almost identical
# to the R (USA) version, except for the model number and ID Block. We
# create and register a class for it, with only the needed overrides
# NOTE: Disabled until detection is fixed
# @directory.register
class FT2900ERadio(FT2900Radio):

"""Yaesu FT-2900E"""
MODEL = "FT-2900E/1900E"
VARIANT = "E"
IDBLOCK = b"\x56\x43\x32\x33\x00\x02\x41\x02\x01\x01"


# This class is for the TX Modified FT-1900/FT-2900 (MARS/CAP Mod).
# Enabling out of band TX changes the header received by CHIRP
@directory.register
class FT2900ModRadio(FT2900Radio):

"""Yaesu FT-2900Mod"""
MODEL = "FT-2900R/1900R(TXMod)"
VARIANT = "Opened Xmit"
IDBLOCK = b"\x56\x43\x32\x33\x00\x02\xc7\x01\x01\x01"

@classmethod
def match_model(cls, filedata, filename):
return False
(3-3/3)