cec: corrected Vizio's vendor id
[deb_libcec.git] / src / LibCecSharp / CecSharpTypes.h
1 #pragma once
2 /*
3 * This file is part of the libCEC(R) library.
4 *
5 * libCEC(R) is Copyright (C) 2011-2012 Pulse-Eight Limited. All rights reserved.
6 * libCEC(R) is an original work, containing original code.
7 *
8 * libCEC(R) is a trademark of Pulse-Eight Limited.
9 *
10 * This program is dual-licensed; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation; either version 2 of the License, or
13 * (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23 *
24 *
25 * Alternatively, you can license this library under a commercial license,
26 * please contact Pulse-Eight Licensing for more information.
27 *
28 * For more information contact:
29 * Pulse-Eight Licensing <license@pulse-eight.com>
30 * http://www.pulse-eight.com/
31 * http://www.pulse-eight.net/
32 */
33
34 #include "../lib/platform/threads/mutex.h"
35 #include <vcclr.h>
36 #include <msclr/marshal.h>
37 #include "../../include/cec.h"
38 #include <vector>
39
40 #using <System.dll>
41
42 namespace CecSharp
43 {
44 public enum class CecDeviceType
45 {
46 Tv = 0,
47 RecordingDevice = 1,
48 Reserved = 2,
49 Tuner = 3,
50 PlaybackDevice = 4,
51 AudioSystem = 5
52 };
53
54 public enum class CecLogLevel
55 {
56 None = 0,
57 Error = 1,
58 Warning = 2,
59 Notice = 4,
60 Traffic = 8,
61 Debug = 16,
62 All = 31
63 };
64
65 public enum class CecLogicalAddress
66 {
67 Unknown = -1, //not a valid logical address
68 Tv = 0,
69 RecordingDevice1 = 1,
70 RecordingDevice2 = 2,
71 Tuner1 = 3,
72 PlaybackDevice1 = 4,
73 AudioSystem = 5,
74 Tuner2 = 6,
75 Tuner3 = 7,
76 PlaybackDevice2 = 8,
77 RecordingDevice3 = 9,
78 Tuner4 = 10,
79 PlaybackDevice3 = 11,
80 Reserved1 = 12,
81 Reserved2 = 13,
82 FreeUse = 14,
83 Unregistered = 15,
84 Broadcast = 15
85 };
86
87 public enum class CecAlert
88 {
89 ServiceDevice = 1
90 };
91
92 public enum class CecParameterType
93 {
94 ParameterTypeString = 1
95 };
96
97 public ref class CecParameter
98 {
99 public:
100 CecParameter(CecParameterType type, System::String ^ strData)
101 {
102 Type = type;
103 Data = strData;
104 }
105
106 property CecParameterType Type;
107 property System::String ^ Data;
108 };
109
110 public enum class CecPowerStatus
111 {
112 On = 0x00,
113 Standby = 0x01,
114 InTransitionStandbyToOn = 0x02,
115 InTransitionOnToStandby = 0x03,
116 Unknown = 0x99
117 };
118
119 public enum class CecVersion
120 {
121 Unknown = 0x00,
122 V1_2 = 0x01,
123 V1_2A = 0x02,
124 V1_3 = 0x03,
125 V1_3A = 0x04,
126 V1_4 = 0x05
127 };
128
129 public enum class CecDisplayControl
130 {
131 DisplayForDefaultTime = 0x00,
132 DisplayUntilCleared = 0x40,
133 ClearPreviousMessage = 0x80,
134 ReservedForFutureUse = 0xC0
135 };
136
137 public enum class CecMenuState
138 {
139 Activated = 0,
140 Deactivated = 1
141 };
142
143 public enum class CecDeckControlMode
144 {
145 SkipForwardWind = 1,
146 SkipReverseRewind = 2,
147 Stop = 3,
148 Eject = 4
149 };
150
151 public enum class CecDeckInfo
152 {
153 Play = 0x11,
154 Record = 0x12,
155 Reverse = 0x13,
156 Still = 0x14,
157 Slow = 0x15,
158 SlowReverse = 0x16,
159 FastForward = 0x17,
160 FastReverse = 0x18,
161 NoMedia = 0x19,
162 Stop = 0x1A,
163 SkipForwardWind = 0x1B,
164 SkipReverseRewind = 0x1C,
165 IndexSearchForward = 0x1D,
166 IndexSearchReverse = 0x1E,
167 OtherStatus = 0x1F
168 };
169
170 public enum class CecUserControlCode
171 {
172 Select = 0x00,
173 Up = 0x01,
174 Down = 0x02,
175 Left = 0x03,
176 Right = 0x04,
177 RightUp = 0x05,
178 RightDown = 0x06,
179 LeftUp = 0x07,
180 LeftDown = 0x08,
181 RootMenu = 0x09,
182 SetupMenu = 0x0A,
183 ContentsMenu = 0x0B,
184 FavoriteMenu = 0x0C,
185 Exit = 0x0D,
186 Number0 = 0x20,
187 Number1 = 0x21,
188 Number2 = 0x22,
189 Number3 = 0x23,
190 Number4 = 0x24,
191 Number5 = 0x25,
192 Number6 = 0x26,
193 Number7 = 0x27,
194 Number8 = 0x28,
195 Number9 = 0x29,
196 Dot = 0x2A,
197 Enter = 0x2B,
198 Clear = 0x2C,
199 NextFavorite = 0x2F,
200 ChannelUp = 0x30,
201 ChannelDown = 0x31,
202 PreviousChannel = 0x32,
203 SoundSelect = 0x33,
204 InputSelect = 0x34,
205 DisplayInformation = 0x35,
206 Help = 0x36,
207 PageUp = 0x37,
208 PageDown = 0x38,
209 Power = 0x40,
210 VolumeUp = 0x41,
211 VolumeDown = 0x42,
212 Mute = 0x43,
213 Play = 0x44,
214 Stop = 0x45,
215 Pause = 0x46,
216 Record = 0x47,
217 Rewind = 0x48,
218 FastForward = 0x49,
219 Eject = 0x4A,
220 Forward = 0x4B,
221 Backward = 0x4C,
222 StopRecord = 0x4D,
223 PauseRecord = 0x4E,
224 Angle = 0x50,
225 SubPicture = 0x51,
226 VideoOnDemand = 0x52,
227 ElectronicProgramGuide = 0x53,
228 TimerProgramming = 0x54,
229 InitialConfiguration = 0x55,
230 PlayFunction = 0x60,
231 PausePlayFunction = 0x61,
232 RecordFunction = 0x62,
233 PauseRecordFunction = 0x63,
234 StopFunction = 0x64,
235 MuteFunction = 0x65,
236 RestoreVolumeFunction = 0x66,
237 TuneFunction = 0x67,
238 SelectMediaFunction = 0x68,
239 SelectAVInputFunction = 0x69,
240 SelectAudioInputFunction = 0x6A,
241 PowerToggleFunction = 0x6B,
242 PowerOffFunction = 0x6C,
243 PowerOnFunction = 0x6D,
244 F1Blue = 0x71,
245 F2Red = 0X72,
246 F3Green = 0x73,
247 F4Yellow = 0x74,
248 F5 = 0x75,
249 Data = 0x76,
250 Max = 0x76,
251 SamsungReturn = 0x91,
252 Unknown
253 };
254
255 public enum class CecVendorId
256 {
257 Samsung = 0x0000F0,
258 LG = 0x00E091,
259 Panasonic = 0x008045,
260 Pioneer = 0x00E036,
261 Onkyo = 0x0009B0,
262 Yamaha = 0x00A0DE,
263 Philips = 0x00903E,
264 Sony = 0x080046,
265 Toshiba = 0x000039,
266 Akai = 0x0020C7,
267 Benq = 0x8065E9,
268 Daewoo = 0x009053,
269 Grundig = 0x00D0D5,
270 Medion = 0x000CB8,
271 Sharp = 0x08001F,
272 Vizio = 0x6B746D,
273 Unknown = 0
274 };
275
276 public enum class CecAudioStatus
277 {
278 MuteStatusMask = 0x80,
279 VolumeStatusMask = 0x7F,
280 VolumeMin = 0x00,
281 VolumeMax = 0x64,
282 VolumeStatusUnknown = 0x7F
283 };
284
285 public enum class CecOpcode
286 {
287 ActiveSource = 0x82,
288 ImageViewOn = 0x04,
289 TextViewOn = 0x0D,
290 InactiveSource = 0x9D,
291 RequestActiveSource = 0x85,
292 RoutingChange = 0x80,
293 RoutingInformation = 0x81,
294 SetStreamPath = 0x86,
295 Standby = 0x36,
296 RecordOff = 0x0B,
297 RecordOn = 0x09,
298 RecordStatus = 0x0A,
299 RecordTvScreen = 0x0F,
300 ClearAnalogueTimer = 0x33,
301 ClearDigitalTimer = 0x99,
302 ClearExternalTimer = 0xA1,
303 SetAnalogueTimer = 0x34,
304 SetDigitalTimer = 0x97,
305 SetExternalTimer = 0xA2,
306 SetTimerProgramTitle = 0x67,
307 TimerClearedStatus = 0x43,
308 TimerStatus = 0x35,
309 CecVersion = 0x9E,
310 GetCecVersion = 0x9F,
311 GivePhysicalAddress = 0x83,
312 GetMenuLanguage = 0x91,
313 ReportPhysicalAddress = 0x84,
314 SetMenuLanguage = 0x32,
315 DeckControl = 0x42,
316 DeckStatus = 0x1B,
317 GiveDeckStatus = 0x1A,
318 Play = 0x41,
319 GiveTunerDeviceStatus = 0x08,
320 SelectAnalogueService = 0x92,
321 SelectDigtalService = 0x93,
322 TunerDeviceStatus = 0x07,
323 TunerStepDecrement = 0x06,
324 TunerStepIncrement = 0x05,
325 DeviceVendorId = 0x87,
326 GiveDeviceVendorId = 0x8C,
327 VendorCommand = 0x89,
328 VendorCommandWithId = 0xA0,
329 VendorRemoteButtonDown = 0x8A,
330 VendorRemoteButtonUp = 0x8B,
331 SetOsdString = 0x64,
332 GiveOsdName = 0x46,
333 SetOsdName = 0x47,
334 MenuRequest = 0x8D,
335 MenuStatus = 0x8E,
336 UserControlPressed = 0x44,
337 UserControlRelease = 0x45,
338 GiveDevicePowerStatus = 0x8F,
339 ReportPowerStatus = 0x90,
340 FeatureAbort = 0x00,
341 Abort = 0xFF,
342 GiveAudioStatus = 0x71,
343 GiveSystemAudioMode = 0x7D,
344 ReportAudioStatus = 0x7A,
345 SetSystemAudioMode = 0x72,
346 SystemAudioModeRequest = 0x70,
347 SystemAudioModeStatus = 0x7E,
348 SetAudioRate = 0x9A,
349 /* when this opcode is set, no opcode will be sent to the device. this is one of the reserved numbers */
350 None = 0xFD
351 };
352
353 public enum class CecSystemAudioStatus
354 {
355 Off = 0,
356 On = 1
357 };
358
359 public enum class CecClientVersion
360 {
361 VersionPre1_5 = 0,
362 Version1_5_0 = 0x1500,
363 Version1_5_1 = 0x1501,
364 Version1_5_2 = 0x1502,
365 Version1_5_3 = 0x1503,
366 Version1_6_0 = 0x1600,
367 Version1_6_1 = 0x1601,
368 Version1_6_2 = 0x1602,
369 Version1_6_3 = 0x1603,
370 Version1_7_0 = 0x1700,
371 Version1_7_1 = 0x1701
372 };
373
374 public enum class CecServerVersion
375 {
376 VersionPre1_5 = 0,
377 Version1_5_0 = 0x1500,
378 Version1_5_1 = 0x1501,
379 Version1_5_2 = 0x1502,
380 Version1_5_3 = 0x1503,
381 Version1_6_0 = 0x1600,
382 Version1_6_1 = 0x1601,
383 Version1_6_2 = 0x1602,
384 Version1_6_3 = 0x1603,
385 Version1_7_0 = 0x1700,
386 Version1_7_1 = 0x1701
387 };
388
389 public ref class CecAdapter
390 {
391 public:
392 CecAdapter(System::String ^ strPath, System::String ^ strComPort)
393 {
394 Path = strPath;
395 ComPort = strComPort;
396 }
397
398 property System::String ^ Path;
399 property System::String ^ ComPort;
400 };
401
402 public ref class CecDeviceTypeList
403 {
404 public:
405 CecDeviceTypeList(void)
406 {
407 Types = gcnew array<CecDeviceType>(5);
408 for (unsigned int iPtr = 0; iPtr < 5; iPtr++)
409 Types[iPtr] = CecDeviceType::Reserved;
410 }
411
412 property array<CecDeviceType> ^ Types;
413 };
414
415 public ref class CecLogicalAddresses
416 {
417 public:
418 CecLogicalAddresses(void)
419 {
420 Addresses = gcnew array<CecLogicalAddress>(16);
421 Clear();
422 }
423
424 void Clear(void)
425 {
426 for (unsigned int iPtr = 0; iPtr < 16; iPtr++)
427 Addresses[iPtr] = CecLogicalAddress::Unknown;
428 }
429
430 bool IsSet(CecLogicalAddress iAddress)
431 {
432 return Addresses[(unsigned int)iAddress] != CecLogicalAddress::Unknown;
433 }
434
435 void Set(CecLogicalAddress iAddress)
436 {
437 Addresses[(unsigned int)iAddress] = iAddress;
438 }
439
440 property CecLogicalAddress Primary;
441 property array<CecLogicalAddress> ^ Addresses;
442 };
443
444 public ref class CecDatapacket
445 {
446 public:
447 CecDatapacket(void)
448 {
449 Data = gcnew array<uint8_t>(100);
450 Size = 0;
451 }
452
453 void PushBack(uint8_t data)
454 {
455 if (Size < 100)
456 {
457 Data[Size] = data;
458 Size++;
459 }
460 }
461
462 property array<uint8_t> ^ Data;
463 property uint8_t Size;
464 };
465
466 public ref class CecCommand
467 {
468 public:
469 CecCommand(CecLogicalAddress iInitiator, CecLogicalAddress iDestination, bool bAck, bool bEom, CecOpcode iOpcode, int32_t iTransmitTimeout)
470 {
471 Initiator = iInitiator;
472 Destination = iDestination;
473 Ack = bAck;
474 Eom = bEom;
475 Opcode = iOpcode;
476 OpcodeSet = true;
477 TransmitTimeout = iTransmitTimeout;
478 Parameters = gcnew CecDatapacket;
479 Empty = false;
480 }
481
482 CecCommand(void)
483 {
484 Initiator = CecLogicalAddress::Unknown;
485 Destination = CecLogicalAddress::Unknown;
486 Ack = false;
487 Eom = false;
488 Opcode = CecOpcode::None;
489 OpcodeSet = false;
490 TransmitTimeout = 0;
491 Parameters = gcnew CecDatapacket;
492 Empty = true;
493 }
494
495 void PushBack(uint8_t data)
496 {
497 if (Initiator == CecLogicalAddress::Unknown && Destination == CecLogicalAddress::Unknown)
498 {
499 Initiator = (CecLogicalAddress) (data >> 4);
500 Destination = (CecLogicalAddress) (data & 0xF);
501 }
502 else if (!OpcodeSet)
503 {
504 OpcodeSet = true;
505 Opcode = (CecOpcode)data;
506 }
507 else
508 {
509 Parameters->PushBack(data);
510 }
511 }
512
513 property bool Empty;
514 property CecLogicalAddress Initiator;
515 property CecLogicalAddress Destination;
516 property bool Ack;
517 property bool Eom;
518 property CecOpcode Opcode;
519 property CecDatapacket ^ Parameters;
520 property bool OpcodeSet;
521 property int32_t TransmitTimeout;
522 };
523
524 public ref class CecKeypress
525 {
526 public:
527 CecKeypress(CecUserControlCode iKeycode, unsigned int iDuration)
528 {
529 Keycode = iKeycode;
530 Duration = iDuration;
531 Empty = false;
532 }
533
534 CecKeypress(void)
535 {
536 Keycode = CecUserControlCode::Unknown;
537 Duration = 0;
538 Empty = true;
539 }
540
541 property bool Empty;
542 property CecUserControlCode Keycode;
543 property unsigned int Duration;
544 };
545
546 public ref class CecLogMessage
547 {
548 public:
549 CecLogMessage(System::String ^ strMessage, CecLogLevel iLevel, int64_t iTime)
550 {
551 Message = strMessage;
552 Level = iLevel;
553 Time = iTime;
554 Empty = false;
555 }
556
557 CecLogMessage(void)
558 {
559 Message = "";
560 Level = CecLogLevel::None;
561 Time = 0;
562 Empty = true;
563 }
564
565 property bool Empty;
566 property System::String ^Message;
567 property CecLogLevel Level;
568 property int64_t Time;
569 };
570
571 ref class CecCallbackMethods; //forward
572 public ref class LibCECConfiguration
573 {
574 public:
575 LibCECConfiguration(void)
576 {
577 DeviceName = "";
578 DeviceTypes = gcnew CecDeviceTypeList();
579 AutodetectAddress = true;
580 PhysicalAddress = CEC_DEFAULT_PHYSICAL_ADDRESS;
581 BaseDevice = (CecLogicalAddress)CEC_DEFAULT_BASE_DEVICE;
582 HDMIPort = CEC_DEFAULT_HDMI_PORT;
583 ClientVersion = CecClientVersion::VersionPre1_5;
584 ServerVersion = CecServerVersion::VersionPre1_5;
585 TvVendor = CecVendorId::Unknown;
586
587 GetSettingsFromROM = false;
588 UseTVMenuLanguage = CEC_DEFAULT_SETTING_USE_TV_MENU_LANGUAGE == 1;
589 ActivateSource = CEC_DEFAULT_SETTING_ACTIVATE_SOURCE == 1;
590
591 WakeDevices = gcnew CecLogicalAddresses();
592 if (CEC_DEFAULT_SETTING_ACTIVATE_SOURCE == 1)
593 WakeDevices->Set(CecLogicalAddress::Tv);
594
595 PowerOffDevices = gcnew CecLogicalAddresses();
596 if (CEC_DEFAULT_SETTING_POWER_OFF_SHUTDOWN == 1)
597 PowerOffDevices->Set(CecLogicalAddress::Broadcast);
598
599 PowerOffScreensaver = CEC_DEFAULT_SETTING_POWER_OFF_SCREENSAVER == 1;
600 PowerOffOnStandby = CEC_DEFAULT_SETTING_POWER_OFF_ON_STANDBY == 1;
601
602 SendInactiveSource = CEC_DEFAULT_SETTING_SEND_INACTIVE_SOURCE == 1;
603 LogicalAddresses = gcnew CecLogicalAddresses();
604 FirmwareVersion = 1;
605 PowerOffDevicesOnStandby = CEC_DEFAULT_SETTING_POWER_OFF_DEVICES_STANDBY == 1;
606 ShutdownOnStandby = CEC_DEFAULT_SETTING_SHUTDOWN_ON_STANDBY == 1;
607 DeviceLanguage = "";
608 }
609
610 void SetCallbacks(CecCallbackMethods ^callbacks)
611 {
612 Callbacks = callbacks;
613 }
614
615 void Update(const CEC::libcec_configuration &config)
616 {
617 DeviceName = gcnew System::String(config.strDeviceName);
618
619 for (unsigned int iPtr = 0; iPtr < 5; iPtr++)
620 DeviceTypes->Types[iPtr] = (CecDeviceType)config.deviceTypes.types[iPtr];
621
622 AutodetectAddress = config.bAutodetectAddress == 1;
623 PhysicalAddress = config.iPhysicalAddress;
624 BaseDevice = (CecLogicalAddress)config.baseDevice;
625 HDMIPort = config.iHDMIPort;
626 ClientVersion = (CecClientVersion)config.clientVersion;
627 ServerVersion = (CecServerVersion)config.serverVersion;
628 TvVendor = (CecVendorId)config.tvVendor;
629
630 // player specific settings
631 GetSettingsFromROM = config.bGetSettingsFromROM == 1;
632 UseTVMenuLanguage = config.bUseTVMenuLanguage == 1;
633 ActivateSource = config.bActivateSource == 1;
634
635 WakeDevices->Clear();
636 for (uint8_t iPtr = 0; iPtr <= 16; iPtr++)
637 if (config.wakeDevices[iPtr])
638 WakeDevices->Set((CecLogicalAddress)iPtr);
639
640 PowerOffDevices->Clear();
641 for (uint8_t iPtr = 0; iPtr <= 16; iPtr++)
642 if (config.powerOffDevices[iPtr])
643 PowerOffDevices->Set((CecLogicalAddress)iPtr);
644
645 PowerOffScreensaver = config.bPowerOffScreensaver == 1;
646 PowerOffOnStandby = config.bPowerOffOnStandby == 1;
647
648 if (ServerVersion >= CecServerVersion::Version1_5_1)
649 SendInactiveSource = config.bSendInactiveSource == 1;
650
651 if (ServerVersion >= CecServerVersion::Version1_5_3)
652 {
653 LogicalAddresses->Clear();
654 for (uint8_t iPtr = 0; iPtr <= 16; iPtr++)
655 if (config.logicalAddresses[iPtr])
656 LogicalAddresses->Set((CecLogicalAddress)iPtr);
657 }
658
659 if (ServerVersion >= CecServerVersion::Version1_6_0)
660 {
661 FirmwareVersion = config.iFirmwareVersion;
662 PowerOffDevicesOnStandby = config.bPowerOffDevicesOnStandby == 1;
663 ShutdownOnStandby = config.bShutdownOnStandby == 1;
664 }
665
666 if (ServerVersion >= CecServerVersion::Version1_6_2)
667 DeviceLanguage = gcnew System::String(config.strDeviceLanguage);
668
669 if (ServerVersion >= CecServerVersion::Version1_6_3)
670 MonitorOnlyClient = config.bMonitorOnly == 1;
671 }
672
673 property System::String ^ DeviceName;
674 property CecDeviceTypeList ^ DeviceTypes;
675 property bool AutodetectAddress;
676 property uint16_t PhysicalAddress;
677 property CecLogicalAddress BaseDevice;
678 property uint8_t HDMIPort;
679 property CecClientVersion ClientVersion;
680 property CecServerVersion ServerVersion;
681 property CecVendorId TvVendor;
682
683 // player specific settings
684 property bool GetSettingsFromROM;
685 property bool UseTVMenuLanguage;
686 property bool ActivateSource;
687 property CecLogicalAddresses ^WakeDevices;
688 property CecLogicalAddresses ^PowerOffDevices;
689 property bool PowerOffScreensaver;
690 property bool PowerOffOnStandby;
691 property bool SendInactiveSource;
692 property CecLogicalAddresses ^LogicalAddresses;
693 property uint16_t FirmwareVersion;
694 property bool PowerOffDevicesOnStandby;
695 property bool ShutdownOnStandby;
696 property bool MonitorOnlyClient;
697 property System::String ^ DeviceLanguage;
698 property CecCallbackMethods ^ Callbacks;
699 };
700
701 // the callback methods are called by unmanaged code, so we need some delegates for this
702 #pragma unmanaged
703 // unmanaged callback methods
704 typedef int (__stdcall *LOGCB) (const CEC::cec_log_message &message);
705 typedef int (__stdcall *KEYCB) (const CEC::cec_keypress &key);
706 typedef int (__stdcall *COMMANDCB)(const CEC::cec_command &command);
707 typedef int (__stdcall *CONFIGCB) (const CEC::libcec_configuration &config);
708 typedef int (__stdcall *ALERTCB) (const CEC::libcec_alert, const CEC::libcec_parameter &data);
709 typedef int (__stdcall *MENUCB) (const CEC::cec_menu_state newVal);
710 typedef void (__stdcall *ACTICB) (const CEC::cec_logical_address logicalAddress, const uint8_t bActivated);
711
712 typedef struct
713 {
714 LOGCB logCB;
715 KEYCB keyCB;
716 COMMANDCB commandCB;
717 CONFIGCB configCB;
718 ALERTCB alertCB;
719 MENUCB menuCB;
720 ACTICB sourceActivatedCB;
721 } UnmanagedCecCallbacks;
722
723 static PLATFORM::CMutex g_callbackMutex;
724 static std::vector<UnmanagedCecCallbacks> g_unmanagedCallbacks;
725 static CEC::ICECCallbacks g_cecCallbacks;
726
727 int CecLogMessageCB(void *cbParam, const CEC::cec_log_message &message)
728 {
729 if (cbParam)
730 {
731 size_t iPtr = (size_t)cbParam;
732 PLATFORM::CLockObject lock(g_callbackMutex);
733 if (iPtr >= 0 && iPtr < g_unmanagedCallbacks.size())
734 return g_unmanagedCallbacks[iPtr].logCB(message);
735 }
736 return 0;
737 }
738
739 int CecKeyPressCB(void *cbParam, const CEC::cec_keypress &key)
740 {
741 if (cbParam)
742 {
743 size_t iPtr = (size_t)cbParam;
744 PLATFORM::CLockObject lock(g_callbackMutex);
745 if (iPtr >= 0 && iPtr < g_unmanagedCallbacks.size())
746 return g_unmanagedCallbacks[iPtr].keyCB(key);
747 }
748 return 0;
749 }
750
751 int CecCommandCB(void *cbParam, const CEC::cec_command &command)
752 {
753 if (cbParam)
754 {
755 size_t iPtr = (size_t)cbParam;
756 PLATFORM::CLockObject lock(g_callbackMutex);
757 if (iPtr >= 0 && iPtr < g_unmanagedCallbacks.size())
758 return g_unmanagedCallbacks[iPtr].commandCB(command);
759 }
760 return 0;
761 }
762
763 int CecConfigCB(void *cbParam, const CEC::libcec_configuration &config)
764 {
765 if (cbParam)
766 {
767 size_t iPtr = (size_t)cbParam;
768 PLATFORM::CLockObject lock(g_callbackMutex);
769 if (iPtr >= 0 && iPtr < g_unmanagedCallbacks.size())
770 return g_unmanagedCallbacks[iPtr].configCB(config);
771 }
772 return 0;
773 }
774
775 int CecAlertCB(void *cbParam, const CEC::libcec_alert alert, const CEC::libcec_parameter &data)
776 {
777 if (cbParam)
778 {
779 size_t iPtr = (size_t)cbParam;
780 PLATFORM::CLockObject lock(g_callbackMutex);
781 if (iPtr >= 0 && iPtr < g_unmanagedCallbacks.size())
782 return g_unmanagedCallbacks[iPtr].alertCB(alert, data);
783 }
784 return 0;
785 }
786
787 int CecMenuCB(void *cbParam, const CEC::cec_menu_state newVal)
788 {
789 if (cbParam)
790 {
791 size_t iPtr = (size_t)cbParam;
792 PLATFORM::CLockObject lock(g_callbackMutex);
793 if (iPtr >= 0 && iPtr < g_unmanagedCallbacks.size())
794 return g_unmanagedCallbacks[iPtr].menuCB(newVal);
795 }
796 return 0;
797 }
798
799 void CecSourceActivatedCB(void *cbParam, const CEC::cec_logical_address logicalAddress, const uint8_t bActivated)
800 {
801 if (cbParam)
802 {
803 size_t iPtr = (size_t)cbParam;
804 PLATFORM::CLockObject lock(g_callbackMutex);
805 if (iPtr >= 0 && iPtr < g_unmanagedCallbacks.size())
806 g_unmanagedCallbacks[iPtr].sourceActivatedCB(logicalAddress, bActivated);
807 }
808 }
809
810 #pragma managed
811 // delegates for the unmanaged callback methods
812 public delegate int CecLogMessageManagedDelegate(const CEC::cec_log_message &);
813 public delegate int CecKeyPressManagedDelegate(const CEC::cec_keypress &);
814 public delegate int CecCommandManagedDelegate(const CEC::cec_command &);
815 public delegate int CecConfigManagedDelegate(const CEC::libcec_configuration &);
816 public delegate int CecAlertManagedDelegate(const CEC::libcec_alert, const CEC::libcec_parameter &);
817 public delegate int CecMenuManagedDelegate(const CEC::cec_menu_state newVal);
818 public delegate void CecSourceActivatedManagedDelegate(const CEC::cec_logical_address logicalAddress, const uint8_t bActivated);
819
820 void AssignCallbacks()
821 {
822 g_cecCallbacks.CBCecLogMessage = CecLogMessageCB;
823 g_cecCallbacks.CBCecKeyPress = CecKeyPressCB;
824 g_cecCallbacks.CBCecCommand = CecCommandCB;
825 g_cecCallbacks.CBCecConfigurationChanged = CecConfigCB;
826 g_cecCallbacks.CBCecAlert = CecAlertCB;
827 g_cecCallbacks.CBCecMenuStateChanged = CecMenuCB;
828 g_cecCallbacks.CBCecSourceActivated = CecSourceActivatedCB;
829 }
830
831 // callback method interface
832 public ref class CecCallbackMethods
833 {
834 public:
835 CecCallbackMethods(void)
836 {
837 m_iCallbackPtr = -1;
838 AssignCallbacks();
839 m_bHasCallbacks = false;
840 m_bDelegatesCreated = false;
841 }
842
843 ~CecCallbackMethods(void)
844 {
845 DestroyDelegates();
846 }
847
848 size_t GetCallbackPtr(void)
849 {
850 PLATFORM::CLockObject lock(g_callbackMutex);
851 return m_iCallbackPtr;
852 }
853
854 protected:
855 !CecCallbackMethods(void)
856 {
857 DestroyDelegates();
858 }
859
860 public:
861 virtual void DisableCallbacks(void)
862 {
863 DestroyDelegates();
864 }
865
866 virtual bool EnableCallbacks(CecCallbackMethods ^ callbacks)
867 {
868 CreateDelegates();
869 if (!m_bHasCallbacks)
870 {
871 m_bHasCallbacks = true;
872 m_callbacks = callbacks;
873 return true;
874 }
875
876 return false;
877 }
878
879 virtual int ReceiveLogMessage(CecLogMessage ^ message)
880 {
881 return 0;
882 }
883
884 virtual int ReceiveKeypress(CecKeypress ^ key)
885 {
886 return 0;
887 }
888
889 virtual int ReceiveCommand(CecCommand ^ command)
890 {
891 return 0;
892 }
893
894 virtual int ConfigurationChanged(LibCECConfiguration ^ config)
895 {
896 return 0;
897 }
898
899 virtual int ReceiveAlert(CecAlert alert, CecParameter ^ data)
900 {
901 return 0;
902 }
903
904 virtual int ReceiveMenuStateChange(CecMenuState newVal)
905 {
906 return 0;
907 }
908
909 virtual void SourceActivated(CecLogicalAddress logicalAddress, bool bActivated)
910 {
911 }
912
913 protected:
914 // managed callback methods
915 int CecLogMessageManaged(const CEC::cec_log_message &message)
916 {
917 int iReturn(0);
918 if (m_bHasCallbacks)
919 iReturn = m_callbacks->ReceiveLogMessage(gcnew CecLogMessage(gcnew System::String(message.message), (CecLogLevel)message.level, message.time));
920 return iReturn;
921 }
922
923 int CecKeyPressManaged(const CEC::cec_keypress &key)
924 {
925 int iReturn(0);
926 if (m_bHasCallbacks)
927 iReturn = m_callbacks->ReceiveKeypress(gcnew CecKeypress((CecUserControlCode)key.keycode, key.duration));
928 return iReturn;
929 }
930
931 int CecCommandManaged(const CEC::cec_command &command)
932 {
933 int iReturn(0);
934 if (m_bHasCallbacks)
935 {
936 CecCommand ^ newCommand = gcnew CecCommand((CecLogicalAddress)command.initiator, (CecLogicalAddress)command.destination, command.ack == 1 ? true : false, command.eom == 1 ? true : false, (CecOpcode)command.opcode, command.transmit_timeout);
937 for (uint8_t iPtr = 0; iPtr < command.parameters.size; iPtr++)
938 newCommand->Parameters->PushBack(command.parameters[iPtr]);
939 iReturn = m_callbacks->ReceiveCommand(newCommand);
940 }
941 return iReturn;
942 }
943
944 int CecConfigManaged(const CEC::libcec_configuration &config)
945 {
946 int iReturn(0);
947 if (m_bHasCallbacks)
948 {
949 LibCECConfiguration ^netConfig = gcnew LibCECConfiguration();
950 netConfig->Update(config);
951 iReturn = m_callbacks->ConfigurationChanged(netConfig);
952 }
953 return iReturn;
954 }
955
956 int CecAlertManaged(const CEC::libcec_alert alert, const CEC::libcec_parameter &data)
957 {
958 int iReturn(0);
959 if (m_bHasCallbacks)
960 {
961 CecParameterType newType = (CecParameterType)data.paramType;
962 if (newType == CecParameterType::ParameterTypeString)
963 {
964 System::String ^ newData = gcnew System::String((const char *)data.paramData, 0, 128);
965 CecParameter ^ newParam = gcnew CecParameter(newType, newData);
966 iReturn = m_callbacks->ReceiveAlert((CecAlert)alert, newParam);
967 }
968 }
969 return iReturn;
970 }
971
972 int CecMenuManaged(const CEC::cec_menu_state newVal)
973 {
974 int iReturn(0);
975 if (m_bHasCallbacks)
976 {
977 iReturn = m_callbacks->ReceiveMenuStateChange((CecMenuState)newVal);
978 }
979 return iReturn;
980 }
981
982 void CecSourceActivatedManaged(const CEC::cec_logical_address logicalAddress, const uint8_t bActivated)
983 {
984 if (m_bHasCallbacks)
985 m_callbacks->SourceActivated((CecLogicalAddress)logicalAddress, bActivated == 1);
986 }
987
988 void DestroyDelegates()
989 {
990 m_bHasCallbacks = false;
991 if (m_bDelegatesCreated)
992 {
993 m_bDelegatesCreated = false;
994 m_logMessageGCHandle.Free();
995 m_keypressGCHandle.Free();
996 m_commandGCHandle.Free();
997 m_alertGCHandle.Free();
998 m_menuGCHandle.Free();
999 m_sourceActivatedGCHandle.Free();
1000 }
1001 }
1002
1003 void CreateDelegates()
1004 {
1005 DestroyDelegates();
1006
1007 if (!m_bDelegatesCreated)
1008 {
1009 msclr::interop::marshal_context ^ context = gcnew msclr::interop::marshal_context();
1010
1011 // create the delegate method for the log message callback
1012 m_logMessageDelegate = gcnew CecLogMessageManagedDelegate(this, &CecCallbackMethods::CecLogMessageManaged);
1013 m_logMessageGCHandle = System::Runtime::InteropServices::GCHandle::Alloc(m_logMessageDelegate);
1014 m_logMessageCallback = static_cast<LOGCB>(System::Runtime::InteropServices::Marshal::GetFunctionPointerForDelegate(m_logMessageDelegate).ToPointer());
1015
1016 // create the delegate method for the keypress callback
1017 m_keypressDelegate = gcnew CecKeyPressManagedDelegate(this, &CecCallbackMethods::CecKeyPressManaged);
1018 m_keypressGCHandle = System::Runtime::InteropServices::GCHandle::Alloc(m_keypressDelegate);
1019 m_keypressCallback = static_cast<KEYCB>(System::Runtime::InteropServices::Marshal::GetFunctionPointerForDelegate(m_keypressDelegate).ToPointer());
1020
1021 // create the delegate method for the command callback
1022 m_commandDelegate = gcnew CecCommandManagedDelegate(this, &CecCallbackMethods::CecCommandManaged);
1023 m_commandGCHandle = System::Runtime::InteropServices::GCHandle::Alloc(m_commandDelegate);
1024 m_commandCallback = static_cast<COMMANDCB>(System::Runtime::InteropServices::Marshal::GetFunctionPointerForDelegate(m_commandDelegate).ToPointer());
1025
1026 // create the delegate method for the configuration change callback
1027 m_configDelegate = gcnew CecConfigManagedDelegate(this, &CecCallbackMethods::CecConfigManaged);
1028 m_configGCHandle = System::Runtime::InteropServices::GCHandle::Alloc(m_configDelegate);
1029 m_configCallback = static_cast<CONFIGCB>(System::Runtime::InteropServices::Marshal::GetFunctionPointerForDelegate(m_configDelegate).ToPointer());
1030
1031 // create the delegate method for the alert callback
1032 m_alertDelegate = gcnew CecAlertManagedDelegate(this, &CecCallbackMethods::CecAlertManaged);
1033 m_alertGCHandle = System::Runtime::InteropServices::GCHandle::Alloc(m_alertDelegate);
1034 m_alertCallback = static_cast<ALERTCB>(System::Runtime::InteropServices::Marshal::GetFunctionPointerForDelegate(m_alertDelegate).ToPointer());
1035
1036 // create the delegate method for the menu callback
1037 m_menuDelegate = gcnew CecMenuManagedDelegate(this, &CecCallbackMethods::CecMenuManaged);
1038 m_menuGCHandle = System::Runtime::InteropServices::GCHandle::Alloc(m_menuDelegate);
1039 m_menuCallback = static_cast<MENUCB>(System::Runtime::InteropServices::Marshal::GetFunctionPointerForDelegate(m_menuDelegate).ToPointer());
1040
1041 // create the delegate method for the source activated callback
1042 m_sourceActivatedDelegate = gcnew CecSourceActivatedManagedDelegate(this, &CecCallbackMethods::CecSourceActivatedManaged);
1043 m_sourceActivatedGCHandle = System::Runtime::InteropServices::GCHandle::Alloc(m_sourceActivatedDelegate);
1044 m_sourceActivatedCallback = static_cast<ACTICB>(System::Runtime::InteropServices::Marshal::GetFunctionPointerForDelegate(m_sourceActivatedDelegate).ToPointer());
1045
1046 delete context;
1047
1048 UnmanagedCecCallbacks unmanagedCallbacks;
1049 unmanagedCallbacks.logCB = m_logMessageCallback;
1050 unmanagedCallbacks.keyCB = m_keypressCallback;
1051 unmanagedCallbacks.commandCB = m_commandCallback;
1052 unmanagedCallbacks.configCB = m_configCallback;
1053 unmanagedCallbacks.alertCB = m_alertCallback;
1054 unmanagedCallbacks.menuCB = m_menuCallback;
1055 unmanagedCallbacks.sourceActivatedCB = m_sourceActivatedCallback;
1056
1057 PLATFORM::CLockObject lock(g_callbackMutex);
1058 g_unmanagedCallbacks.push_back(unmanagedCallbacks);
1059 m_iCallbackPtr = g_unmanagedCallbacks.size() - 1;
1060 m_bDelegatesCreated = true;
1061 }
1062 }
1063
1064 CecLogMessageManagedDelegate ^ m_logMessageDelegate;
1065 static System::Runtime::InteropServices::GCHandle m_logMessageGCHandle;
1066 LOGCB m_logMessageCallback;
1067
1068 CecKeyPressManagedDelegate ^ m_keypressDelegate;
1069 static System::Runtime::InteropServices::GCHandle m_keypressGCHandle;
1070 KEYCB m_keypressCallback;
1071
1072 CecCommandManagedDelegate ^ m_commandDelegate;
1073 static System::Runtime::InteropServices::GCHandle m_commandGCHandle;
1074 COMMANDCB m_commandCallback;
1075
1076 CecConfigManagedDelegate ^ m_configDelegate;
1077 static System::Runtime::InteropServices::GCHandle m_configGCHandle;
1078 CONFIGCB m_configCallback;
1079
1080 CecAlertManagedDelegate ^ m_alertDelegate;
1081 static System::Runtime::InteropServices::GCHandle m_alertGCHandle;
1082 ALERTCB m_alertCallback;
1083
1084 CecMenuManagedDelegate ^ m_menuDelegate;
1085 static System::Runtime::InteropServices::GCHandle m_menuGCHandle;
1086 MENUCB m_menuCallback;
1087
1088 CecSourceActivatedManagedDelegate ^ m_sourceActivatedDelegate;
1089 static System::Runtime::InteropServices::GCHandle m_sourceActivatedGCHandle;
1090 ACTICB m_sourceActivatedCallback;
1091
1092 CecCallbackMethods ^ m_callbacks;
1093 bool m_bHasCallbacks;
1094 bool m_bDelegatesCreated;
1095 size_t m_iCallbackPtr;
1096 };
1097 }