001package jmri.jmrix.loconet; 002 003import jmri.InstanceManager; 004import jmri.Programmer; 005import jmri.ProgrammingMode; 006import jmri.beans.PropertyChangeSupport; 007import jmri.jmrit.decoderdefn.DecoderFile; 008import jmri.jmrit.decoderdefn.DecoderIndexFile; 009import jmri.jmrit.roster.Roster; 010import jmri.jmrit.roster.RosterEntry; 011import jmri.jmrix.ProgrammingTool; 012import jmri.jmrix.loconet.lnsvf1.Lnsv1Device; 013import jmri.jmrix.loconet.lnsvf1.Lnsv1Devices; 014import jmri.jmrix.loconet.lnsvf1.Lnsv1MessageContents; 015import jmri.managers.DefaultProgrammerManager; 016//import jmri.progdebugger.ProgDebugger; 017 018import jmri.util.ThreadingUtil; 019import jmri.util.swing.JmriJOptionPane; 020 021import javax.annotation.concurrent.GuardedBy; 022import java.util.List; 023 024/** 025 * LocoNet LNSV1 Devices Manager 026 * <p> 027 * A centralized resource to help identify LocoNet "LNSV1 Format" 028 * devices and "manage" them. 029 * <p> 030 * Supports the following features: 031 * - LNSV1 "discovery" process supported via BROADCAST call 032 * - LNSV1 Device "destination address" change supported by writing a new value to LNSV 0 (close session next) 033 * - LNSV1 Device "reconfigure/reset" not supported/documented 034 * - identification of devices with conflicting "destination address"es (warning before program start) 035 * - identification of a matching JMRI "decoder definition" for each discovered 036 * device, if an appropriate definition exists (only 1 value is matched, checks for LNSVf1 protocol support) 037 * - identification of matching JMRI "roster entry" which matches each 038 * discovered device, if an appropriate roster entry exists 039 * - ability to open a symbolic programmer for a given discovered device, if 040 * an appropriate roster entry exists 041 * 042 * @author B. Milhaupt Copyright (c) 2020 043 * @author Egbert Broerse (c) 2021, 2025 044 */ 045 046public class Lnsv1DevicesManager extends PropertyChangeSupport 047 implements LocoNetListener, jmri.Disposable { 048 private final LocoNetSystemConnectionMemo memo; 049 @GuardedBy("this") 050 private final Lnsv1Devices lnsv1Devices; 051 052 // constant for thread name, with memo prefix appended. 053 static final String ROSTER_THREAD_NAME = "rosterMatchingListLnsv1DM"; 054 055 public Lnsv1DevicesManager(@javax.annotation.Nonnull LocoNetSystemConnectionMemo memo) { 056 this.memo = memo; 057 if (memo.getLnTrafficController() != null) { 058 memo.getLnTrafficController().addLocoNetListener(~0, this); 059 } else { 060 log.error("No LocoNet connection available, this tool cannot function"); // NOI18N 061 } 062 synchronized (this) { 063 lnsv1Devices = new Lnsv1Devices(); 064 } 065 } 066 067 public synchronized Lnsv1Devices getDeviceList() { 068 return lnsv1Devices; 069 } 070 071 public synchronized int getDeviceCount() { 072 return lnsv1Devices.size(); 073 } 074 075 public void clearDevicesList() { 076 synchronized (this) { 077 lnsv1Devices.removeAllDevices(); 078 } 079 jmri.util.ThreadingUtil.runOnLayoutEventually( ()-> firePropertyChange("DeviceListChanged", true, false)); 080 } 081 082 /** 083 * Extract module information from a LNSVf1 READ_ONE REPLY message. 084 * If not already in the lnsv1Devices list, try to find a matching decoder definition (by address number and programming mode) 085 * and add it. Skip if already in the list. 086 * 087 * @param m The received LocoNet message. Note that this same object may 088 * be presented to multiple users. It should not be modified 089 * here. 090 */ 091 @Override 092 public void message(LocoNetMessage m) { 093 if (Lnsv1MessageContents.isSupportedSv1Message(m)) { 094 if ((Lnsv1MessageContents.extractMessageType(m) == Lnsv1MessageContents.Sv1Command.SV1_READ) && 095 (Lnsv1MessageContents.extractMessageVersion(m) > 0)) { // which marks replies from devices 096 // it's an LNSV1 Read_One Reply message, decode contents: 097 Lnsv1MessageContents contents = new Lnsv1MessageContents(m); 098 int vrs = contents.getVersionNum(); 099 int addrL = contents.getSrcL(); 100 int subAddr = contents.getSubAddress(); 101 int sv = contents.getSvNum(); 102 int val = contents.getSvValue(); 103 log.debug("Lnsv1DevicesManager got read reply: vrs:{}, address:{}/{} cv:{} val:{}", vrs, addrL, subAddr, sv, val); 104 105 synchronized (this) { 106 if (lnsv1Devices.addDevice(new Lnsv1Device(addrL, subAddr, sv, val, "", "", vrs))) { 107 log.debug("new Lnsv1Device added to table"); 108 // Annotate the discovered device LNSV1 data based on address 109 for (int i = 0; i < lnsv1Devices.size(); ++i) { // find the added item 110 Lnsv1Device dev = lnsv1Devices.getDevice(i); 111 if ((dev.getDestAddrLow() == addrL) && (dev.getDestAddrHigh() == subAddr)) { 112 // Try to find a roster entry which matches the device characteristics 113 log.debug("Looking for adr {} in Roster", dev.getDestAddr()); 114 115 // threadUtil off GUI for Roster reading decoderfiles cf. LncvDevicesManager 116 ThreadingUtil.newThread(() -> { 117 List<RosterEntry> rl; 118 try { 119 // requires nonnull default for jmri.jmrit.roster.RosterConfigManager 120 rl = Roster.getDefault().getEntriesMatchingCriteria( 121 Integer.toString(dev.getDestAddr()), // composite DCC address 122 null, null, null, 123 null); // TODO filter on progMode LNSV1 only on new roster entries 124 log.debug("Lnsv1DeviceManager found {} matches in Roster", rl.size()); 125 if (rl.isEmpty()) { 126 log.debug("No corresponding roster entry found"); 127 } else if (rl.size() == 1) { 128 log.debug("Matching roster entry found"); 129 dev.setRosterEntry(rl.get(0)); // link this device to the entry 130 String title = rl.get(0).getDecoderModel() + " (" + rl.get(0).getDecoderFamily() + ")"; 131 // fileFromTitle() matches by model + " (" + family + ")" 132 DecoderFile decoderFile = InstanceManager.getDefault(DecoderIndexFile.class).fileFromTitle(title); 133 if (decoderFile != null) { 134 // TODO check for LNSV1 mode 135 dev.setDecoderFile(decoderFile); // link to decoderFile (to check programming mode from table) 136 log.debug("Attached a decoderfile"); 137 } else { 138 log.warn("Could not attach decoderfile {} to entry", rl.get(0).getFileName()); 139 } 140 } else { // matches > 1 141 JmriJOptionPane.showMessageDialog(null, 142 Bundle.getMessage("WarnMultipleLnsv1ModsFound", rl.size(), addrL, subAddr), 143 Bundle.getMessage("WarningTitle"), JmriJOptionPane.WARNING_MESSAGE); 144 log.info("Found multiple matching LNSV1 roster entries. " + "Cannot associate any one to this device."); 145 } 146 147 } catch (Exception e) { 148 log.error("Error creating Roster.matchingList: {}", e.getMessage()); 149 } 150 }, ROSTER_THREAD_NAME + memo.getSystemPrefix()).start(); 151 // this will block until the thread completes, either by finishing or by being cancelled 152 153 // notify listeners of pertinent change to device list 154 firePropertyChange("DeviceListChanged", true, false); 155 } 156 } 157 } else { 158 log.debug("LNSV1 device was already in list"); 159 } 160 } 161 } else { 162 log.debug("LNSV1 message not a READ REPLY [{}]", m); 163 } 164 } else { 165 log.debug("LNSV1 message not recognized"); 166 } 167 } 168 169 public synchronized Lnsv1Device getDevice(int vrs, int addr) { 170 for (int i = 0; i < lnsv1Devices.size(); ++ i) { 171 Lnsv1Device dev = lnsv1Devices.getDevice(i); 172 if ((dev.getSwVersion() == vrs) && (dev.getDestAddr() == addr)) { 173 return dev; 174 } 175 } 176 return null; 177 } 178 179 public ProgrammingResult prepareForSymbolicProgrammer(Lnsv1Device dev, ProgrammingTool t) { 180 synchronized(this) { 181 if (lnsv1Devices.isDeviceExistant(dev) < 0) { 182 return ProgrammingResult.FAIL_NO_SUCH_DEVICE; 183 } 184 int destAddr = dev.getDestAddr(); 185 if (destAddr == 0) { 186 return ProgrammingResult.FAIL_DESTINATION_ADDRESS_IS_ZERO; 187 } 188 int deviceCount = 0; 189 for (Lnsv1Device d : lnsv1Devices.getDevices()) { 190 if (destAddr == d.getDestAddr()) { 191 deviceCount++; 192 } 193 } 194 log.debug("prepareForSymbolicProgrammer found {} matches", deviceCount); 195 if (deviceCount > 1) { 196 return ProgrammingResult.FAIL_MULTIPLE_DEVICES_SAME_DESTINATION_ADDRESS; 197 } 198 } 199 200 if ((dev.getRosterName() == null) || (dev.getRosterName().isEmpty())) { 201 return ProgrammingResult.FAIL_NO_MATCHING_ROSTER_ENTRY; 202 } 203 204 // check if roster entry still present in Roster 205 RosterEntry re = Roster.getDefault().entryFromTitle(dev.getRosterName()); 206 if (re == null) { 207 log.warn("Could not open LNSV1 Programmer because {} not found in Roster. Removed from device", 208 dev.getRosterName()); 209 dev.setRosterEntry(null); 210 jmri.util.ThreadingUtil.runOnLayoutEventually( ()-> firePropertyChange("DeviceListChanged", true, false)); 211 return ProgrammingResult.FAIL_NO_MATCHING_ROSTER_ENTRY; 212 } 213 String name = re.getId(); 214 215 DefaultProgrammerManager pm = memo.getProgrammerManager(); 216 if (pm == null) { 217 return ProgrammingResult.FAIL_NO_APPROPRIATE_PROGRAMMER; 218 } 219 Programmer p = pm.getAddressedProgrammer(false, dev.getDestAddr()); 220 if (p == null) { 221 return ProgrammingResult.FAIL_NO_ADDRESSED_PROGRAMMER; 222 } 223 224 //if (p.getClass() != ProgDebugger.class) { // Debug in Simulator 225 // ProgDebugger is used for LocoNet HexFile Sim; uncommenting above line allows testing of LNSV1 Tool 226 if (!p.getSupportedModes().contains(LnProgrammerManager.LOCONETOPSBOARD)) { 227 return ProgrammingResult.FAIL_NO_LNSV1_PROGRAMMER; 228 } 229 p.setMode(LnProgrammerManager.LOCONETSV1MODE); 230 ProgrammingMode prgMode = p.getMode(); 231 if (!prgMode.equals(LnProgrammerManager.LOCONETSV1MODE)) { 232 return ProgrammingResult.FAIL_NO_LNSV1_PROGRAMMER; 233 } 234 //} 235 236 t.openPaneOpsProgFrame(re, name, "programmers/Comprehensive.xml", p); // NOI18N 237 return ProgrammingResult.SUCCESS_PROGRAMMER_OPENED; 238 } 239 240 @Override 241 public void dispose(){ 242 if (memo.getLnTrafficController() != null) { 243 memo.getLnTrafficController().removeLocoNetListener(~0, this); 244 } 245 } 246 247 public enum ProgrammingResult { 248 SUCCESS_PROGRAMMER_OPENED, 249 FAIL_NO_SUCH_DEVICE, 250 FAIL_NO_APPROPRIATE_PROGRAMMER, 251 FAIL_NO_MATCHING_ROSTER_ENTRY, 252 FAIL_DESTINATION_ADDRESS_IS_ZERO, 253 FAIL_MULTIPLE_DEVICES_SAME_DESTINATION_ADDRESS, 254 FAIL_NO_ADDRESSED_PROGRAMMER, 255 FAIL_NO_LNSV1_PROGRAMMER 256 } 257 258 private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(Lnsv1DevicesManager.class); 259 260}