LibCecSharp: delete the internal callback struct when closing the connection
[deb_libcec.git] / src / LibCecSharp / LibCecSharp.cpp
index 92a4eee84e18116a847b11bae513b238013941a1..971dc4d6f7daea3d3370b127ade3a2ec6d8d0f47 100644 (file)
@@ -43,32 +43,79 @@ namespace CecSharp
        public ref class LibCecSharp : public CecCallbackMethods
        {
        public:
+         LibCecSharp(LibCECConfiguration ^config)
+               {
+                       CecCallbackMethods::EnableCallbacks(config->Callbacks);
+                       if (!InitialiseLibCec(config))
+                               throw gcnew Exception("Could not initialise LibCecSharp");
+               }
+
                LibCecSharp(String ^ strDeviceName, CecDeviceTypeList ^ deviceTypes)
+               {
+                       LibCECConfiguration ^config = gcnew LibCECConfiguration();
+                       config->SetCallbacks(this);
+                       config->DeviceName  = strDeviceName;
+                       config->DeviceTypes = deviceTypes;
+                       if (!InitialiseLibCec(config))
+                               throw gcnew Exception("Could not initialise LibCecSharp");
+               }
+          
+               ~LibCecSharp(void)
+               {
+                       Close();
+                       m_libCec = NULL;
+               }
+
+       private:
+               !LibCecSharp(void)
+               {
+                       Close();
+                       m_libCec = NULL;
+               }
+
+               bool InitialiseLibCec(LibCECConfiguration ^config)
                {
                        marshal_context ^ context = gcnew marshal_context();
-                       m_bHasCallbacks = false;
-                       const char* strDeviceNameC = context->marshal_as<const char*>(strDeviceName);
+                       libcec_configuration libCecConfig;
+                       ConvertConfiguration(context, config, libCecConfig);
 
-                       cec_device_type_list types;
-                       for (unsigned int iPtr = 0; iPtr < 5; iPtr++)
-                               types.types[iPtr] = (cec_device_type)deviceTypes->Types[iPtr];
-                       m_libCec = (ICECAdapter *) CECInit(strDeviceNameC, types);
+                       m_libCec = (ICECAdapter *) CECInitialise(&libCecConfig);
 
                        delete context;
+                       return m_libCec != NULL;
+               }
+
+         void ConvertConfiguration(marshal_context ^context, LibCECConfiguration ^netConfig, CEC::libcec_configuration &config)
+               {
+                       config.Clear();
+
+                       _snprintf_s(config.strDeviceName, 13, context->marshal_as<const char*>(netConfig->DeviceName));
+                       for (unsigned int iPtr = 0; iPtr < 5; iPtr++)
+                               config.deviceTypes.types[iPtr] = (cec_device_type)netConfig->DeviceTypes->Types[iPtr];
+
+                       config.iPhysicalAddress     = netConfig->PhysicalAddress;
+                       config.baseDevice           = (cec_logical_address)netConfig->BaseDevice;
+                       config.iHDMIPort            = netConfig->HDMIPort;
+                       config.clientVersion        = (cec_client_version)netConfig->ClientVersion;
+                       config.bGetSettingsFromROM  = netConfig->GetSettingsFromROM;
+                       config.bActivateSource      = netConfig->ActivateSource;
+                       config.tvVendor             = (cec_vendor_id)netConfig->TvVendor;
+                       config.wakeDevices.Clear();
+                       for (int iPtr = 0; iPtr < 16; iPtr++)
+                       {
+                               if (netConfig->WakeDevices->IsSet((CecLogicalAddress)iPtr))
+                                       config.wakeDevices.Set((cec_logical_address)iPtr);
+                       }
+                       config.powerOffDevices.Clear();
+                       for (int iPtr = 0; iPtr < 16; iPtr++)
+                       {
+                               if (netConfig->PowerOffDevices->IsSet((CecLogicalAddress)iPtr))
+                                       config.powerOffDevices.Set((cec_logical_address)iPtr);
+                       }
+                       config.bPowerOffScreensaver = netConfig->PowerOffScreensaver;
+                       config.bPowerOffOnStandby   = netConfig->PowerOffOnStandby;
+                       config.callbacks            = &g_cecCallbacks;
                }
-          
-                ~LibCecSharp(void)
-                {
-                        Close();
-                        m_libCec = NULL;
-                }
-
-       protected:
-                !LibCecSharp(void)
-                {
-                        Close();
-                        m_libCec = NULL;
-                }
 
        public:
                array<CecAdapter ^> ^ FindAdapters(String ^ path)
@@ -100,6 +147,10 @@ namespace CecSharp
 
                void Close(void)
                {
+                       // delete the callbacks, since these might already have been destroyed in .NET
+                       CecCallbackMethods::DisableCallbacks();
+                       m_libCec->EnableCallbacks(NULL, NULL);
+                       m_libCec->StandbyDevices();
                        m_libCec->Close();
                }
 
@@ -190,7 +241,7 @@ namespace CecSharp
                        return m_libCec->SetLogicalAddress((cec_logical_address) logicalAddress);
                }
 
-               bool SetPhysicalAddress(int16_t physicalAddress)
+               bool SetPhysicalAddress(uint16_t physicalAddress)
                {
                        return m_libCec->SetPhysicalAddress(physicalAddress);
                }
@@ -352,6 +403,91 @@ namespace CecSharp
                        return m_libCec->GetDevicePhysicalAddress((cec_logical_address)iAddress);
                }
 
+               bool SetStreamPath(CecLogicalAddress iAddress)
+               {
+                       return m_libCec->SetStreamPath((cec_logical_address)iAddress);
+               }
+
+               bool SetStreamPath(uint16_t iPhysicalAddress)
+               {
+                       return m_libCec->SetStreamPath(iPhysicalAddress);
+               }
+
+               CecLogicalAddresses ^GetLogicalAddresses(void)
+               {
+                       CecLogicalAddresses ^addr = gcnew CecLogicalAddresses();
+                       cec_logical_addresses libAddr = m_libCec->GetLogicalAddresses();
+                       for (unsigned int iPtr = 0; iPtr < 16; iPtr++)
+                               addr->Addresses[iPtr] = (CecLogicalAddress)libAddr.addresses[iPtr];
+                       addr->Primary = (CecLogicalAddress)libAddr.primary;
+                       return addr;
+               }
+
+               bool GetCurrentConfiguration(LibCECConfiguration ^configuration)
+               {
+                       libcec_configuration config;
+                       config.Clear();
+
+                       if (m_libCec->GetCurrentConfiguration(&config))
+                       {
+                               configuration->BaseDevice = (CecLogicalAddress)config.baseDevice;
+                               configuration->DeviceName = gcnew String(config.strDeviceName);
+                               configuration->HDMIPort = config.iHDMIPort;
+                               configuration->PhysicalAddress = config.iPhysicalAddress;
+                               configuration->PowerOffOnStandby = config.bPowerOffOnStandby == 1;
+                               configuration->PowerOffScreensaver = config.bPowerOffScreensaver == 1;
+                               configuration->ActivateSource = config.bActivateSource == 1;
+                               configuration->TvVendor = (CecVendorId)config.tvVendor;
+
+                               configuration->WakeDevices->Clear();
+                               int iDevices(0);
+                               for (uint8_t iPtr = 0; iPtr <= 16; iPtr++)
+                                       if (config.wakeDevices[iPtr])
+                                               configuration->WakeDevices->Addresses[iDevices++] = (CecLogicalAddress)iPtr;
+
+                               configuration->PowerOffDevices->Clear();
+                               iDevices = 0;
+                               for (uint8_t iPtr = 0; iPtr <= 16; iPtr++)
+                                       if (config.powerOffDevices[iPtr])
+                                               configuration->PowerOffDevices->Addresses[iDevices++] = (CecLogicalAddress)iPtr;
+
+                               configuration->UseTVMenuLanguage = config.bUseTVMenuLanguage == 1;
+                               for (unsigned int iPtr = 0; iPtr < 5; iPtr++)
+                                       configuration->DeviceTypes->Types[iPtr] = (CecDeviceType)config.deviceTypes.types[iPtr];
+                               return true;
+                       }
+                       return false;
+               }
+
+    bool CanPersistConfiguration(void)
+               {
+                       return m_libCec->CanPersistConfiguration();
+               }
+
+    bool PersistConfiguration(LibCECConfiguration ^configuration)
+               {
+                       marshal_context ^ context = gcnew marshal_context();
+                       libcec_configuration config;
+                       ConvertConfiguration(context, configuration, config);
+
+                       bool bReturn = m_libCec->PersistConfiguration(&config);
+
+                       delete context;
+                       return bReturn;
+               }
+
+               bool SetConfiguration(LibCECConfiguration ^configuration)
+               {
+                       marshal_context ^ context = gcnew marshal_context();
+                       libcec_configuration config;
+                       ConvertConfiguration(context, configuration, config);
+
+                       bool bReturn = m_libCec->SetConfiguration(&config);
+
+                       delete context;
+                       return bReturn;
+               }
+
                String ^ ToString(CecLogicalAddress iAddress)
                {
                        const char *retVal = m_libCec->ToString((cec_logical_address)iAddress);