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