MythTV master
firewiredevice.cpp
Go to the documentation of this file.
1
7// C++ headers
8#include <algorithm>
9#include <chrono> // for milliseconds
10#include <thread> // for sleep_for
11
12// Qt headers
13#include <QMap>
14
15// MythTV headers
16#include "libmythbase/mythconfig.h"
18
19#include "linuxfirewiredevice.h"
20#if CONFIG_FIREWIRE_OSX
22#endif
23#include "mpeg/mpegtables.h"
24
25#define LOC QString("FireDev(%1): ").arg(guid_to_string(m_guid))
26
27static void fw_init(QMap<uint64_t,QString> &id_to_model);
28
29QMap<uint64_t,QString> FirewireDevice::s_idToModel;
31
32FirewireDevice::FirewireDevice(uint64_t guid, uint subunitid, uint speed) :
33 m_guid(guid), m_subunitid(subunitid),
34 m_speed(speed)
35{
36}
37
39{
40 if (listener)
41 {
42 auto it = find(m_listeners.begin(), m_listeners.end(), listener);
43 if (it == m_listeners.end())
44 m_listeners.push_back(listener);
45 }
46
47 LOG(VB_RECORD, LOG_INFO, LOC +
48 QString("AddListener() %1").arg(m_listeners.size()));
49}
50
52{
53 auto it = find(m_listeners.begin(), m_listeners.end(), listener);
54 while (it != m_listeners.end())
55 {
56 it = m_listeners.erase(it);
57 it = find(it, m_listeners.end(), listener);
58 }
59
60 LOG(VB_RECORD, LOG_INFO, LOC +
61 QString("RemoveListener() %1").arg(m_listeners.size()));
62}
63
65{
66 QMutexLocker locker(&m_lock);
67
68 std::vector<uint8_t> cmd;
69 std::vector<uint8_t> ret;
70
71 cmd.push_back(kAVCControlCommand);
73 cmd.push_back(kAVCUnitPowerOpcode);
74 cmd.push_back((on) ? kAVCPowerStateOn : kAVCPowerStateOff);
75
76 QString cmdStr = (on) ? "on" : "off";
77 LOG(VB_RECORD, LOG_INFO, LOC + QString("Powering %1").arg(cmdStr));
78
79 if (!SendAVCCommand(cmd, ret, -1))
80 {
81 LOG(VB_GENERAL, LOG_ERR, LOC + "Power on cmd failed (no response)");
82 return false;
83 }
84
85 if (kAVCAcceptedStatus != ret[0])
86 {
87 LOG(VB_GENERAL, LOG_ERR, LOC + QString("Power %1 failed").arg(cmdStr));
88
89 return false;
90 }
91
92 LOG(VB_RECORD, LOG_INFO, LOC +
93 QString("Power %1 cmd sent successfully").arg(cmdStr));
94
95 return true;
96}
97
99{
100 QMutexLocker locker(&m_lock);
101
102 std::vector<uint8_t> cmd;
103 std::vector<uint8_t> ret;
104
105 cmd.push_back(kAVCStatusInquiryCommand);
107 cmd.push_back(kAVCUnitPowerOpcode);
108 cmd.push_back(kAVCPowerStateQuery);
109
110 LOG(VB_CHANNEL, LOG_INFO, LOC + "Requesting STB Power State");
111
112 if (!SendAVCCommand(cmd, ret, -1))
113 {
114 LOG(VB_GENERAL, LOG_ERR, LOC + "Power cmd failed (no response)");
116 }
117
118 QString loc = LOC + "STB Power State: ";
119
120 if (ret[0] != kAVCResponseImplemented)
121 {
122 LOG(VB_CHANNEL, LOG_INFO, loc + "Query not implemented");
123 return kAVCPowerUnknown;
124 }
125
126 // check 1st operand..
127 if (ret[3] == kAVCPowerStateOn)
128 {
129 LOG(VB_CHANNEL, LOG_INFO, loc + "On");
130 return kAVCPowerOn;
131 }
132
133 if (ret[3] == kAVCPowerStateOff)
134 {
135 LOG(VB_CHANNEL, LOG_INFO, loc + "Off");
136 return kAVCPowerOff;
137 }
138
139 LOG(VB_GENERAL, LOG_ERR, LOC + "STB Power State: Unknown Response");
140
141 return kAVCPowerUnknown;
142}
143
144bool FirewireDevice::SetChannel(const QString &panel_model,
145 uint alt_method, uint channel)
146{
147 LOG(VB_CHANNEL, LOG_INFO, QString("SetChannel(model %1, alt %2, chan %3)")
148 .arg(panel_model).arg(alt_method).arg(channel));
149
150 QMutexLocker locker(&m_lock);
151 LOG(VB_CHANNEL, LOG_INFO, "SetChannel() -- locked");
152
153 if (!IsSTBSupported(panel_model))
154 {
155 LOG(VB_GENERAL, LOG_ERR, LOC +
156 QString("Model: '%1' ").arg(panel_model) +
157 "is not supported by internal channel changer.");
158 return false;
159 }
160
161 std::array<uint,3> digit {
162 (channel % 1000) / 100,
163 (channel % 100) / 10,
164 (channel % 10)
165 };
166
168 {
169 LOG(VB_GENERAL, LOG_ERR, LOC +
170 "SetChannel: Extended subunits are not supported.");
171
172 return false;
173 }
174
175 std::vector<uint8_t> cmd;
176 std::vector<uint8_t> ret;
177
178 if ((panel_model.toUpper() == "SA GENERIC") ||
179 (panel_model.toUpper() == "SA4200HD") ||
180 (panel_model.toUpper() == "SA4250HDC"))
181 {
182 if (panel_model.toUpper() == "SA4250HDC")
183 {
184 LOG(VB_GENERAL, LOG_ERR, LOC +
185 "The Scientific Atlanta 4250 HDC is not supported "
186 "\n\t\t\tby any MythTV Firewire channel changer."
187 "At the moment you must use an IR blaster.");
188 }
189
190 cmd.push_back(kAVCControlCommand);
191 cmd.push_back(kAVCSubunitTypePanel | m_subunitid);
192 cmd.push_back(kAVCPanelPassThrough);
194
195 cmd.push_back(4); // operand length
196 cmd.push_back((channel>>8) & 0x0f);
197 cmd.push_back(channel & 0xff);
198 cmd.push_back(0x00);
199 cmd.push_back(0x00);
200
201 if (!SendAVCCommand(cmd, ret, -1))
202 return false;
203
204 bool press_ok = (kAVCAcceptedStatus == ret[0]);
205
207 if (!SendAVCCommand(cmd, ret, -1))
208 return false;
209
210 bool release_ok = (kAVCAcceptedStatus == ret[0]);
211
212 if (!press_ok && !release_ok)
213 {
214 LOG(VB_GENERAL, LOG_ERR, LOC + "Tuning failed");
215 return false;
216 }
217
218 SetLastChannel(channel);
219 return true;
220 }
221
222 // the PACE is obviously not a Motorola channel changer, but the
223 // same commands work for it as the Motorola.
224 bool is_mot = ((panel_model.startsWith("DCT-", Qt::CaseInsensitive)) ||
225 (panel_model.startsWith("DCH-", Qt::CaseInsensitive)) ||
226 (panel_model.startsWith("DCX-", Qt::CaseInsensitive)) ||
227 (panel_model.startsWith("QIP-", Qt::CaseInsensitive)) ||
228 (panel_model.startsWith("MOTO", Qt::CaseInsensitive)) ||
229 (panel_model.startsWith("PACE-", Qt::CaseInsensitive)));
230
231 if (is_mot && !alt_method)
232 {
233 for (uint d : digit)
234 {
235 cmd.clear();
236 cmd.push_back(kAVCControlCommand);
237 cmd.push_back(kAVCSubunitTypePanel | m_subunitid);
238 cmd.push_back(kAVCPanelPassThrough);
239 cmd.push_back((kAVCPanelKey0 + d) | kAVCPanelKeyPress);
240 cmd.push_back(0x00);
241 cmd.push_back(0x00);
242 cmd.push_back(0x00);
243 cmd.push_back(0x00);
244
245 if (!SendAVCCommand(cmd, ret, -1))
246 return false;
247
248 std::this_thread::sleep_for(500ms);
249 }
250
251 SetLastChannel(channel);
252 return true;
253 }
254
255 if (is_mot && alt_method)
256 {
257 cmd.push_back(kAVCControlCommand);
258 cmd.push_back(kAVCSubunitTypePanel | m_subunitid);
259 cmd.push_back(kAVCPanelPassThrough);
261
262 cmd.push_back(4); // operand length
263 cmd.push_back((channel>>8) & 0x0f);
264 cmd.push_back(channel & 0xff);
265 cmd.push_back(0x00);
266 cmd.push_back(0xff);
267
268 if (!SendAVCCommand(cmd, ret, -1))
269 return false;
270
271 SetLastChannel(channel);
272 return true;
273 }
274
275 if (panel_model.toUpper() == "SA3250HD")
276 {
277 cmd.push_back(kAVCControlCommand);
278 cmd.push_back(kAVCSubunitTypePanel | m_subunitid);
279 cmd.push_back(kAVCPanelPassThrough);
281
282 cmd.push_back(4); // operand length
283 cmd.push_back(0x30 | digit[2]);
284 cmd.push_back(0x30 | digit[1]);
285 cmd.push_back(0x30 | digit[0]);
286 cmd.push_back(0xff);
287
288 if (!SendAVCCommand(cmd, ret, -1))
289 return false;
290
291 cmd[5] = 0x30 | digit[0];
292 cmd[6] = 0x30 | digit[1];
293 cmd[7] = 0x30 | digit[2];
294
295 if (!SendAVCCommand(cmd, ret, -1))
296 return false;
297
298 SetLastChannel(channel);
299 return true;
300 }
301
302 return false;
303}
304
306 const unsigned char *data, uint dataSize)
307{
308 if ((dataSize >= TSPacket::kSize) && (data[0] == SYNC_BYTE) &&
309 ((data[1] & 0x1f) == 0) && (data[2] == 0))
310 {
311 ProcessPATPacket(*(reinterpret_cast<const TSPacket*>(data)));
312 }
313
314 for (auto & listener : m_listeners)
315 listener->AddData(data, dataSize);
316}
317
319{
320 m_bufferCleared = (channel == m_lastChannel);
321 m_lastChannel = channel;
322
323 LOG(VB_GENERAL, LOG_INFO, QString("SetLastChannel(%1): cleared: %2")
324 .arg(channel).arg(m_bufferCleared ? "yes" : "no"));
325}
326
328{
329 if (!tspacket.TransportError() && !tspacket.Scrambled() &&
330 tspacket.HasPayload() && tspacket.PayloadStart() && (tspacket.PID() == 0))
331 {
332 PSIPTable pes(tspacket);
333 uint crc = pes.CalcCRC();
334 m_bufferCleared |= (crc != m_lastCrc);
335 m_lastCrc = crc;
336#if 0
337 LOG(VB_RECORD, LOG_DEBUG, LOC +
338 QString("ProcessPATPacket: CRC 0x%1 cleared: %2")
339 .arg(crc,0,16).arg(m_bufferCleared ? "yes" : "no"));
340#endif
341 }
342 else
343 {
344 LOG(VB_GENERAL, LOG_ERR, LOC + "Can't handle large PAT's");
345 }
346}
347
348QString FirewireDevice::GetModelName(uint vendor_id, uint model_id)
349{
350 QMutexLocker locker(&s_staticLock);
351 if (s_idToModel.empty())
353
354 QString ret = s_idToModel[(((uint64_t) vendor_id) << 32) | model_id];
355
356 if (ret.isEmpty())
357 return "MOTO GENERIC";
358 return ret;
359}
360
361std::vector<AVCInfo> FirewireDevice::GetSTBList(void)
362{
363 std::vector<AVCInfo> list;
364
365#if CONFIG_FIREWIRE_LINUX
367#elif CONFIG_FIREWIRE_OSX
369#endif
370
371// #define DEBUG_AVC_INFO
372#ifdef DEBUG_AVC_INFO
374 info.m_guid = 0x0016928a7b600001ULL;
375 info.m_specid = 0x0;
376 info.m_vendorid = 0x000014f8;
377 info.m_modelid = 0x00001072;
378 info.m_firmware_revision = 0x0;
379 info.m_product_name = "Explorer 4200 HD";
380 list.push_back(info);
381
382 info.m_guid = 0xff2145a850e39810ULL;
383 info.m_specid = 0x0;
384 info.m_vendorid = 0x000014f8;
385 info.m_modelid = 0x00000be0;
386 info.m_firmware_revision = 0x0;
387 info.m_product_name = "Explorer 3250 HD";
388 list.push_back(info);
389#endif // DEBUG_AVC_INFO
390
391 return list;
392}
393
394static void fw_init(QMap<uint64_t,QString> &id_to_model)
395{
396 const std::array<const uint64_t,16> sa_vendor_ids
397 {
398 0x0a73, 0x0f21, 0x11e6, 0x14f8, 0x1692, 0x1868,
399 0x1947, 0x1ac3, 0x1bd7, 0x1cea, 0x1e6b, 0x21be,
400 0x223a, 0x22ce, 0x23be, 0x252e,
401 };
402
403 for (uint64_t vendor_id : sa_vendor_ids)
404 {
405 id_to_model[vendor_id << 32 | 0x0be0] = "SA3250HD";
406 id_to_model[vendor_id << 32 | 0x1072] = "SA4200HD";
407 id_to_model[vendor_id << 32 | 0x10cc] = "SA4250HDC";
408 id_to_model[vendor_id << 32 | 0x22ce] = "SA8300HD";
409 }
410
411 const std::array<uint64_t,59> motorola_vendor_ids
412 {
413 /* DCH-3200, DCX-3200 */
414 0x1c11, 0x1cfb, 0x1fc4, 0x23a3, 0x23ee, 0x25f1,
415 0xfa01, 0x25f1, 0x25f2, 0xcc7d37, 0x946269, 0x6455b1,
416 /* DCX-3432 */
417 0x24a0,
418 /* DCH-3416 */
419 0x1e46,
420 /* DCT-3416 */
421 0x1bdd,
422 /* DCT-3412 */
423 0x159a,
424 /* DCT-6200, DCT-3416 */
425 0x0ce5, 0x0e5c, 0x1225, 0x0f9f, 0x1180,
426 0x12c9, 0x11ae, 0x152f, 0x14e8, 0x16b5, 0x1371,
427 0x19a6, 0x1aad, 0x0b06, 0x195e, 0x10dc,
428 /* DCT-6212 */
429 0x0f9f, 0x152f,
430 /* DCT-6216, 2224 */
431 0x17ee, 0x1a66,
432 /* QIP 6200 */
433 0x211e,
434 /* QIP 7100 */
435 0x2374,
436 /* unknown, see http://standards.ieee.org/regauth/oui/oui.txt */
437 0x04db, 0x0406, 0x0ce5, 0x111a, 0x1225, 0x1404,
438 0x1626, 0x18c0, 0x1ade, 0x1cfb, 0x2040, 0x2180,
439 0x2210, 0x230b, 0x2375, 0x2395, 0x23a2, 0x23ed,
440 0x23ee, 0x23a0, 0x23a1,
441 };
442
443 for (uint64_t vendor_id : motorola_vendor_ids)
444 {
445 id_to_model[vendor_id << 32 | 0xf740] = "DCX-3200";
446 id_to_model[vendor_id << 32 | 0xf804] = "DCX-3200";
447 id_to_model[vendor_id << 32 | 0xfa03] = "DCX-3200";
448 id_to_model[vendor_id << 32 | 0xfa05] = "DCX-3200";
449 id_to_model[vendor_id << 32 | 0xfa07] = "DCX-3200";
450 id_to_model[vendor_id << 32 | 0x24a1] = "DCX-3200";
451 id_to_model[vendor_id << 32 | 0x2322] = "DCX-3200";
452 id_to_model[vendor_id << 32 | 0xea05] = "DCX-3432";
453 id_to_model[vendor_id << 32 | 0xd330] = "DCH-3200";
454 id_to_model[vendor_id << 32 | 0xb630] = "DCH-3416";
455 id_to_model[vendor_id << 32 | 0x34cb] = "DCT-3412";
456 id_to_model[vendor_id << 32 | 0x346b] = "DCT-3416";
457 id_to_model[vendor_id << 32 | 0xb630] = "DCT-3416";
458 id_to_model[vendor_id << 32 | 0x6200] = "DCT-6200";
459 id_to_model[vendor_id << 32 | 0x620a] = "DCT-6200";
460 id_to_model[vendor_id << 32 | 0x64ca] = "DCT-6212";
461 id_to_model[vendor_id << 32 | 0x64cb] = "DCT-6212";
462 id_to_model[vendor_id << 32 | 0x646b] = "DCT-6216";
463 id_to_model[vendor_id << 32 | 0x8100] = "QIP-7100";
464 id_to_model[vendor_id << 32 | 0x7100] = "QIP-6200";
465 id_to_model[vendor_id << 32 | 0x0001] = "QIP-7100";
466 }
467
468 const std::array<const uint64_t,2> pace_vendor_ids
469 {
470 /* PACE 550-HD & 779 */
471 0x1cc3, 0x5094,
472 };
473
474 for (uint64_t vendor_id : pace_vendor_ids)
475 {
476 id_to_model[vendor_id << 32 | 0x10551] = "PACE-550";
477 id_to_model[vendor_id << 32 | 0x10755] = "PACE-779";
478 }
479}
480
481bool FirewireDevice::IsSTBSupported(const QString &panel_model)
482{
483 QString model = panel_model.toUpper();
484 return ((model == "DCH-3200") ||
485 (model == "DCH-3416") ||
486 (model == "DCT-3412") ||
487 (model == "DCT-3416") ||
488 (model == "DCT-6200") ||
489 (model == "DCT-6212") ||
490 (model == "DCT-6216") ||
491 (model == "DCX-3200") ||
492 (model == "SA3250HD") ||
493 (model == "SA4200HD") ||
494 (model == "SA4250HDC") ||
495 (model == "SA8300HD") ||
496 (model == "PACE-550") ||
497 (model == "PACE-779") ||
498 (model == "QIP-6200") ||
499 (model == "QIP-7100") ||
500 (model == "SA GENERIC") ||
501 (model == "MOTO GENERIC"));
502}
static std::vector< AVCInfo > GetSTBList(void)
virtual void AddListener(TSDataListener *listener)
void ProcessPATPacket(const TSPacket &tspacket)
static QMap< uint64_t, QString > s_idToModel
Vendor ID + Model ID to FirewireDevice STB model string.
virtual void RemoveListener(TSDataListener *listener)
void SetLastChannel(uint channel)
static QMutex s_staticLock
FirewireDevice(uint64_t guid, uint subunitid, uint speed)
virtual PowerState GetPowerState(void)
static std::vector< AVCInfo > GetSTBList(void)
std::vector< TSDataListener * > m_listeners
virtual bool SendAVCCommand(const std::vector< uint8_t > &cmd, std::vector< uint8_t > &result, int retry_cnt)=0
virtual bool SetChannel(const QString &panel_model, uint alt_method, uint channel)
static bool IsSTBSupported(const QString &model)
static QString GetModelName(uint vendor_id, uint model_id)
virtual bool SetPowerState(bool on)
virtual void BroadcastToListeners(const unsigned char *data, uint dataSize)
static std::vector< AVCInfo > GetSTBList(void)
uint CalcCRC(void) const
Definition: pespacket.cpp:172
A PSIP table is a variant of a PES packet containing an MPEG, ATSC or DVB table.
Definition: mpegtables.h:410
unsigned int PID(void) const
Definition: tspacket.h:93
bool HasPayload(void) const
Definition: tspacket.h:116
bool PayloadStart(void) const
Definition: tspacket.h:89
bool TransportError(void) const
Definition: tspacket.h:86
bool Scrambled(void) const
Definition: tspacket.h:112
Used to access the data of a Transport Stream packet.
Definition: tspacket.h:208
static constexpr unsigned int kSize
Definition: tspacket.h:261
unsigned int uint
Definition: compat.h:60
static pid_list_t::iterator find(const PIDInfoMap &map, pid_list_t &list, pid_list_t::iterator begin, pid_list_t::iterator end, bool find_open)
#define LOC
FirewireDevice Copyright (c) 2005 by Jim Westfall Distributed as part of MythTV under GPL v2 and late...
static void fw_init(QMap< uint64_t, QString > &id_to_model)
static const iso6937table * d
#define LOG(_MASK_, _LEVEL_, _QSTRING_)
Definition: mythlogging.h:39
dictionary info
Definition: azlyrics.py:7
static constexpr uint8_t SYNC_BYTE
Definition: tspacket.h:21