diff --git a/src/platform/Tizen/BUILD.gn b/src/platform/Tizen/BUILD.gn index c30f733107de57..6c3a046ad052d1 100644 --- a/src/platform/Tizen/BUILD.gn +++ b/src/platform/Tizen/BUILD.gn @@ -52,6 +52,8 @@ static_library("Tizen") { "PosixConfig.h", "SystemPlatformConfig.h", "SystemTimeSupport.cpp", + "WiFiManager.cpp", + "WiFiManager.h", ] deps = [ "${chip_root}/src/setup_payload" ] diff --git a/src/platform/Tizen/CHIPDevicePlatformConfig.h b/src/platform/Tizen/CHIPDevicePlatformConfig.h index dac49181253831..685887004defa9 100644 --- a/src/platform/Tizen/CHIPDevicePlatformConfig.h +++ b/src/platform/Tizen/CHIPDevicePlatformConfig.h @@ -26,7 +26,7 @@ // ==================== Platform Adaptations ==================== -#define CHIP_DEVICE_CONFIG_ENABLE_WIFI_STATION 0 +#define CHIP_DEVICE_CONFIG_ENABLE_WIFI_STATION 1 #define CHIP_DEVICE_CONFIG_ENABLE_WIFI_AP 0 #define CHIP_DEVICE_CONFIG_ENABLE_THREAD 0 diff --git a/src/platform/Tizen/ConnectivityManagerImpl.cpp b/src/platform/Tizen/ConnectivityManagerImpl.cpp index 0e869e567b1052..c9f27c41c08360 100644 --- a/src/platform/Tizen/ConnectivityManagerImpl.cpp +++ b/src/platform/Tizen/ConnectivityManagerImpl.cpp @@ -51,6 +51,18 @@ ConnectivityManagerImpl ConnectivityManagerImpl::sInstance; CHIP_ERROR ConnectivityManagerImpl::_Init(void) { CHIP_ERROR err = CHIP_NO_ERROR; + + mWiFiStationMode = kWiFiStationMode_Disabled; + mWiFiAPMode = kWiFiAPMode_Disabled; + mWiFiAPState = kWiFiAPState_NotActive; + mLastAPDemandTime = 0; + mWiFiStationReconnectIntervalMS = CHIP_DEVICE_CONFIG_WIFI_STATION_RECONNECT_INTERVAL; + mWiFiAPIdleTimeoutMS = CHIP_DEVICE_CONFIG_WIFI_AP_IDLE_TIMEOUT; + +#if CHIP_DEVICE_CONFIG_ENABLE_WIFI + WiFiMgr().Init(); +#endif + return err; } @@ -60,56 +72,121 @@ void ConnectivityManagerImpl::_OnPlatformEvent(const ChipDeviceEvent * event) {} ConnectivityManager::WiFiStationMode ConnectivityManagerImpl::_GetWiFiStationMode(void) { - return ConnectivityManager::kWiFiStationMode_NotSupported; + CHIP_ERROR err = CHIP_NO_ERROR; + wifi_manager_device_state_e deviceState = WIFI_MANAGER_DEVICE_STATE_DEACTIVATED; + + ReturnErrorCodeIf(mWiFiStationMode == kWiFiStationMode_ApplicationControlled, mWiFiStationMode); + + err = WiFiMgr().GetDeviceState(&deviceState); + VerifyOrReturnError(err == CHIP_NO_ERROR, mWiFiStationMode); + + mWiFiStationMode = (deviceState == WIFI_MANAGER_DEVICE_STATE_ACTIVATED) ? kWiFiStationMode_Enabled : kWiFiStationMode_Disabled; + + return mWiFiStationMode; } CHIP_ERROR ConnectivityManagerImpl::_SetWiFiStationMode(ConnectivityManager::WiFiStationMode val) { - return CHIP_ERROR_NOT_IMPLEMENTED; + CHIP_ERROR err = CHIP_NO_ERROR; + wifi_manager_device_state_e deviceState = WIFI_MANAGER_DEVICE_STATE_DEACTIVATED; + + ReturnErrorCodeIf(val == kWiFiStationMode_NotSupported, CHIP_ERROR_INVALID_ARGUMENT); + + if (val != kWiFiStationMode_ApplicationControlled) + { + deviceState = + (val == kWiFiStationMode_Disabled) ? WIFI_MANAGER_DEVICE_STATE_DEACTIVATED : WIFI_MANAGER_DEVICE_STATE_ACTIVATED; + err = WiFiMgr().SetDeviceState(deviceState); + VerifyOrReturnError(err == CHIP_NO_ERROR, err); + } + + if (mWiFiStationMode != val) + { + ChipLogProgress(DeviceLayer, "WiFi station mode change: %s -> %s", WiFiStationModeToStr(mWiFiStationMode), + WiFiStationModeToStr(val)); + + mWiFiStationMode = val; + } + + return err; } uint32_t ConnectivityManagerImpl::_GetWiFiStationReconnectIntervalMS(void) { - return 0; + return mWiFiStationReconnectIntervalMS; } CHIP_ERROR ConnectivityManagerImpl::_SetWiFiStationReconnectIntervalMS(uint32_t val) { - return CHIP_ERROR_NOT_IMPLEMENTED; + mWiFiStationReconnectIntervalMS = val; + + return CHIP_NO_ERROR; } bool ConnectivityManagerImpl::_IsWiFiStationEnabled(void) { - return false; + bool isWifiStationEnabled = false; + + WiFiMgr().IsActivated(&isWifiStationEnabled); + + return isWifiStationEnabled; } bool ConnectivityManagerImpl::_IsWiFiStationConnected(void) { - return false; -} + CHIP_ERROR err = CHIP_NO_ERROR; + wifi_manager_connection_state_e connectionState = WIFI_MANAGER_CONNECTION_STATE_DISCONNECTED; + bool isWifiStationConnected = false; -bool ConnectivityManagerImpl::_IsWiFiStationApplicationControlled(void) -{ - return false; + err = WiFiMgr().GetConnectionState(&connectionState); + VerifyOrReturnError(err == CHIP_NO_ERROR, isWifiStationConnected); + + if (connectionState == WIFI_MANAGER_CONNECTION_STATE_CONNECTED) + isWifiStationConnected = true; + + return isWifiStationConnected; } bool ConnectivityManagerImpl::_IsWiFiStationProvisioned(void) { - return false; + CHIP_ERROR err = CHIP_NO_ERROR; + wifi_manager_connection_state_e connectionState = WIFI_MANAGER_CONNECTION_STATE_DISCONNECTED; + bool isWifiStationProvisioned = false; + + err = WiFiMgr().GetConnectionState(&connectionState); + VerifyOrReturnError(err == CHIP_NO_ERROR, isWifiStationProvisioned); + + if (connectionState >= WIFI_MANAGER_CONNECTION_STATE_ASSOCIATION) + isWifiStationProvisioned = true; + + return isWifiStationProvisioned; } -void ConnectivityManagerImpl::_ClearWiFiStationProvision(void) {} +void ConnectivityManagerImpl::_ClearWiFiStationProvision(void) +{ + WiFiMgr().RemoveAllConfigs(); +} bool ConnectivityManagerImpl::_CanStartWiFiScan(void) { return false; } +ConnectivityManager::WiFiAPMode ConnectivityManagerImpl::_GetWiFiAPMode() +{ + return mWiFiAPMode; +} + CHIP_ERROR ConnectivityManagerImpl::_SetWiFiAPMode(WiFiAPMode val) { return CHIP_ERROR_NOT_IMPLEMENTED; } +bool ConnectivityManagerImpl::_IsWiFiAPActive() +{ + return mWiFiAPState == kWiFiAPState_Active; +} + void ConnectivityManagerImpl::_DemandStartWiFiAP(void) {} void ConnectivityManagerImpl::_StopOnDemandWiFiAP(void) {} @@ -117,11 +194,31 @@ void ConnectivityManagerImpl::_StopOnDemandWiFiAP(void) {} void ConnectivityManagerImpl::_MaintainOnDemandWiFiAP(void) {} void ConnectivityManagerImpl::_SetWiFiAPIdleTimeoutMS(uint32_t val) {} + +void ConnectivityManagerImpl::StartWiFiManagement(void) +{ + SystemLayer().ScheduleWork(ActivateWiFiManager, NULL); +} + +void ConnectivityManagerImpl::StopWiFiManagement(void) +{ + SystemLayer().ScheduleWork(DeactivateWiFiManager, NULL); +} + +void ConnectivityManagerImpl::ActivateWiFiManager(::chip::System::Layer * aLayer, void * aAppState) +{ + WiFiMgr().Activate(); +} + +void ConnectivityManagerImpl::DeactivateWiFiManager(::chip::System::Layer * aLayer, void * aAppState) +{ + WiFiMgr().Deactivate(); +} #endif // CHIP_DEVICE_CONFIG_ENABLE_WIFI CHIP_ERROR ConnectivityManagerImpl::ProvisionWiFiNetwork(const char * ssid, const char * key) { - return CHIP_ERROR_NOT_IMPLEMENTED; + return WiFiMgr().Connect(ssid, key); } } // namespace DeviceLayer diff --git a/src/platform/Tizen/ConnectivityManagerImpl.h b/src/platform/Tizen/ConnectivityManagerImpl.h index 1175600fee22ef..6eabba52a0f997 100644 --- a/src/platform/Tizen/ConnectivityManagerImpl.h +++ b/src/platform/Tizen/ConnectivityManagerImpl.h @@ -37,6 +37,10 @@ #include #endif +#if CHIP_DEVICE_CONFIG_ENABLE_WIFI +#include "WiFiManager.h" +#endif + namespace chip { namespace Inet { class IPAddress; @@ -74,6 +78,11 @@ class ConnectivityManagerImpl final : public ConnectivityManager, public: CHIP_ERROR ProvisionWiFiNetwork(const char * ssid, const char * key); +#if CHIP_DEVICE_CONFIG_ENABLE_WIFI + void StartWiFiManagement(void); + void StopWiFiManagement(void); +#endif + private: // ===== Members that implement the ConnectivityManager abstract interface. @@ -101,6 +110,9 @@ class ConnectivityManagerImpl final : public ConnectivityManager, void _MaintainOnDemandWiFiAP(void); uint32_t _GetWiFiAPIdleTimeoutMS(void); void _SetWiFiAPIdleTimeoutMS(uint32_t val); + + static void ActivateWiFiManager(::chip::System::Layer * aLayer, void * aAppState); + static void DeactivateWiFiManager(::chip::System::Layer * aLayer, void * aAppState); #endif // ===== Members for internal use by the following friends. @@ -121,12 +133,7 @@ class ConnectivityManagerImpl final : public ConnectivityManager, }; #if CHIP_DEVICE_CONFIG_ENABLE_WIFI -inline ConnectivityManager::WiFiAPMode ConnectivityManagerImpl::_GetWiFiAPMode() -{ - return ConnectivityManager::kWiFiAPMode_NotSupported; -} - -inline bool ConnectivityManagerImpl::_IsWiFiAPActive() +inline bool ConnectivityManagerImpl::_IsWiFiStationApplicationControlled(void) { return false; } diff --git a/src/platform/Tizen/DeviceNetworkProvisioningDelegateImpl.cpp b/src/platform/Tizen/DeviceNetworkProvisioningDelegateImpl.cpp index 21f0c0fe3c9741..423a5b030a9f70 100644 --- a/src/platform/Tizen/DeviceNetworkProvisioningDelegateImpl.cpp +++ b/src/platform/Tizen/DeviceNetworkProvisioningDelegateImpl.cpp @@ -25,7 +25,17 @@ namespace DeviceLayer { CHIP_ERROR DeviceNetworkProvisioningDelegateImpl::_ProvisionWiFiNetwork(const char * ssid, const char * key) { - return CHIP_ERROR_NOT_IMPLEMENTED; + CHIP_ERROR err = CHIP_NO_ERROR; + + ChipLogProgress(NetworkProvisioning, "TizenNetworkProvisioningDelegate: SSID: %s", ssid); + + err = ConnectivityMgrImpl().ProvisionWiFiNetwork(ssid, key); + if (err != CHIP_NO_ERROR) + { + ChipLogError(NetworkProvisioning, "Failed to connect to WiFi network: %s", chip::ErrorStr(err)); + } + + return err; } } // namespace DeviceLayer diff --git a/src/platform/Tizen/WiFiManager.cpp b/src/platform/Tizen/WiFiManager.cpp new file mode 100644 index 00000000000000..8a06d7a0e7336f --- /dev/null +++ b/src/platform/Tizen/WiFiManager.cpp @@ -0,0 +1,824 @@ +/* + * + * Copyright (c) 2021 Project CHIP Authors + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include + +#if CHIP_DEVICE_CONFIG_ENABLE_WIFI +#include "MainLoop.h" +#include "WiFiManager.h" + +namespace { +const char * __WiFiDeviceStateToStr(wifi_manager_device_state_e state) +{ + switch (state) + { + case WIFI_MANAGER_DEVICE_STATE_DEACTIVATED: + return "Deactivated"; + case WIFI_MANAGER_DEVICE_STATE_ACTIVATED: + return "Activated"; + default: + return "(unknown)"; + } +} + +const char * __WiFiScanStateToStr(wifi_manager_scan_state_e state) +{ + switch (state) + { + case WIFI_MANAGER_SCAN_STATE_NOT_SCANNING: + return "Not scanning"; + case WIFI_MANAGER_SCAN_STATE_SCANNING: + return "Scanning"; + default: + return "(unknown)"; + } +} + +const char * __WiFiConnectionStateToStr(wifi_manager_connection_state_e state) +{ + switch (state) + { + case WIFI_MANAGER_CONNECTION_STATE_FAILURE: + return "Failure"; + case WIFI_MANAGER_CONNECTION_STATE_DISCONNECTED: + return "Disconnected"; + case WIFI_MANAGER_CONNECTION_STATE_ASSOCIATION: + return "Association"; + case WIFI_MANAGER_CONNECTION_STATE_CONNECTED: + return "Connected"; + case WIFI_MANAGER_CONNECTION_STATE_CONFIGURATION: + return "Configuration"; + default: + return "(unknown)"; + } +} + +const char * __WiFiIPConflictStateToStr(wifi_manager_ip_conflict_state_e state) +{ + switch (state) + { + case WIFI_MANAGER_IP_CONFLICT_STATE_CONFLICT_NOT_DETECTED: + return "Removed"; + case WIFI_MANAGER_IP_CONFLICT_STATE_CONFLICT_DETECTED: + return "Detacted"; + default: + return "(unknown)"; + } +} + +const char * __WiFiModuleStateToStr(wifi_manager_module_state_e state) +{ + switch (state) + { + case WIFI_MANAGER_MODULE_STATE_DETACHED: + return "Detached"; + case WIFI_MANAGER_MODULE_STATE_ATTACHED: + return "Attached"; + default: + return "(unknown)"; + } +} + +const char * __WiFiSecurityTypeToStr(wifi_manager_security_type_e type) +{ + switch (type) + { + case WIFI_MANAGER_SECURITY_TYPE_NONE: + return "None"; + case WIFI_MANAGER_SECURITY_TYPE_WEP: + return "WEP"; + case WIFI_MANAGER_SECURITY_TYPE_WPA_PSK: + return "WPA"; + case WIFI_MANAGER_SECURITY_TYPE_WPA2_PSK: + return "WPA2"; + case WIFI_MANAGER_SECURITY_TYPE_EAP: + return "EAP"; + case WIFI_MANAGER_SECURITY_TYPE_WPA_FT_PSK: + return "FT_PSK"; + case WIFI_MANAGER_SECURITY_TYPE_SAE: + return "WPA3"; + case WIFI_MANAGER_SECURITY_TYPE_OWE: + return "OWE"; + case WIFI_MANAGER_SECURITY_TYPE_DPP: + return "DPP"; + default: + return "(unknown)"; + } +} +} // namespace + +namespace chip { +namespace DeviceLayer { +namespace Internal { + +WiFiManager WiFiManager::sInstance; + +void WiFiManager::_DeviceStateChangedCb(wifi_manager_device_state_e deviceState, void * userData) +{ + ChipLogProgress(DeviceLayer, "WiFi device state changed [%s]", __WiFiDeviceStateToStr(deviceState)); + sInstance._WiFiSetDeviceState(deviceState); +} + +void WiFiManager::_ModuleStateChangedCb(wifi_manager_module_state_e moduleState, void * userData) +{ + ChipLogProgress(DeviceLayer, "WiFi module state changed [%s]", __WiFiModuleStateToStr(moduleState)); + sInstance._WiFiSetModuleState(moduleState); +} + +void WiFiManager::_ConnectionStateChangedCb(wifi_manager_connection_state_e connectionState, wifi_manager_ap_h ap, void * userData) +{ + ChipLogProgress(DeviceLayer, "WiFi connection state changed [%s]", __WiFiConnectionStateToStr(connectionState)); + sInstance._WiFiSetConnectionState(connectionState); +} + +void WiFiManager::_ScanStateChangedCb(wifi_manager_scan_state_e scanState, void * userData) +{ + ChipLogProgress(DeviceLayer, "WiFi scan state changed [%s]", __WiFiScanStateToStr(scanState)); +} + +void WiFiManager::_RssiLevelChangedCb(wifi_manager_rssi_level_e rssiLevel, void * userData) +{ + ChipLogProgress(DeviceLayer, "WiFi rssi level changed [%d]", rssiLevel); +} + +void WiFiManager::_BackgroundScanCb(wifi_manager_error_e wifiErr, void * userData) +{ + ChipLogProgress(DeviceLayer, "WiFi background scan completed [%s]", get_error_message(wifiErr)); +} + +void WiFiManager::_IPConflictCb(char * mac, wifi_manager_ip_conflict_state_e ipConflictState, void * userData) +{ + ChipLogProgress(DeviceLayer, "WiFi ip conflict [%s %s]", mac, __WiFiIPConflictStateToStr(ipConflictState)); +} + +void WiFiManager::_ActivateCb(wifi_manager_error_e wifiErr, void * userData) +{ + GMainLoop * loop = (GMainLoop *) userData; + + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi is activated"); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: activate WiFi [%s]", get_error_message(wifiErr)); + } + + g_main_loop_quit(loop); +} + +void WiFiManager::_DeactivateCb(wifi_manager_error_e wifiErr, void * userData) +{ + GMainLoop * loop = (GMainLoop *) userData; + + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi is deactivated"); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: deactivate WiFi [%s]", get_error_message(wifiErr)); + } + + g_main_loop_quit(loop); +} + +void WiFiManager::_ScanFinishedCb(wifi_manager_error_e wifiErr, void * userData) +{ + GMainLoop * loop = (GMainLoop *) userData; + wifi_manager_ap_h foundAp = NULL; + + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi scan finished"); + + foundAp = sInstance._WiFiGetFoundAP(); + if (foundAp != NULL) + { + MainLoop::Instance().AsyncRequest(_WiFiConnect, static_cast(foundAp)); + } + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: scan WiFi [%s]", get_error_message(wifiErr)); + } + + g_main_loop_quit(loop); +} + +bool WiFiManager::_FoundAPCb(wifi_manager_ap_h ap, void * userData) +{ + bool cbRet = true; + int wifiErr = WIFI_MANAGER_ERROR_NONE; + char * essid = NULL; + bool isPassphraseRequired = false; + wifi_manager_ap_h * clonedAp = (wifi_manager_ap_h *) userData; + + wifiErr = wifi_manager_ap_get_essid(ap, &essid); + VerifyOrExit(wifiErr == WIFI_MANAGER_ERROR_NONE, + ChipLogProgress(DeviceLayer, "FAIL: get AP essid [%s]", get_error_message(wifiErr))); + + VerifyOrExit(strcmp(sInstance.mWiFiSSID, essid) == 0, ); + + wifiErr = wifi_manager_ap_is_passphrase_required(ap, &isPassphraseRequired); + VerifyOrExit(wifiErr == WIFI_MANAGER_ERROR_NONE, + ChipLogProgress(DeviceLayer, "FAIL: get AP passphrase required [%s]", get_error_message(wifiErr))); + + if (isPassphraseRequired) + { + wifiErr = wifi_manager_ap_set_passphrase(ap, sInstance.mWiFiKey); + VerifyOrExit(wifiErr == WIFI_MANAGER_ERROR_NONE, + ChipLogProgress(DeviceLayer, "FAIL: set AP passphrase [%s]", get_error_message(wifiErr))); + } + + wifiErr = wifi_manager_ap_clone(clonedAp, ap); + VerifyOrExit(wifiErr == WIFI_MANAGER_ERROR_NONE, + ChipLogProgress(DeviceLayer, "FAIL: clone AP [%s]", get_error_message(wifiErr))); + + cbRet = false; + +exit: + memset(sInstance.mWiFiSSID, 0, sizeof(sInstance.mWiFiSSID)); + memset(sInstance.mWiFiKey, 0, sizeof(sInstance.mWiFiKey)); + g_free(essid); + return cbRet; +} + +void WiFiManager::_ConnectedCb(wifi_manager_error_e wifiErr, void * userData) +{ + GMainLoop * loop = (GMainLoop *) userData; + + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi is connected"); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: connect WiFi [%s]", get_error_message(wifiErr)); + } + + g_main_loop_quit(loop); +} + +bool WiFiManager::_ConfigListCb(const wifi_manager_config_h config, void * userData) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + char * name = NULL; + wifi_manager_security_type_e securityType = WIFI_MANAGER_SECURITY_TYPE_NONE; + + wifi_manager_config_get_name(config, &name); + wifi_manager_config_get_security_type(config, &securityType); + + wifiErr = wifi_manager_config_remove(sInstance.mWiFiManagerHandle, config); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Remove config [%s:%s]", name, __WiFiSecurityTypeToStr(securityType)); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: remove config [%s]", get_error_message(wifiErr)); + } + + g_free(name); + return true; +} + +gboolean WiFiManager::_WiFiInitialize(gpointer userData) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + wifiErr = wifi_manager_initialize(&(sInstance.mWiFiManagerHandle)); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi is initialized"); + sInstance._WiFiSetStates(); + sInstance._WiFiSetCallbacks(); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: initialize WiFi [%s]", get_error_message(wifiErr)); + return false; + } + + return true; +} + +void WiFiManager::_WiFiDeinitialize(void) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + _WiFiUnsetCallbacks(); + + wifiErr = wifi_manager_deinitialize(mWiFiManagerHandle); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi is deinitialized"); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: deinitialize WiFi [%s]", get_error_message(wifiErr)); + } +} + +gboolean WiFiManager::_WiFiActivate(GMainLoop * mainLoop, gpointer userData) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + wifiErr = wifi_manager_activate(sInstance.mWiFiManagerHandle, _ActivateCb, mainLoop); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi activation is requested"); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: request WiFi activation [%s]", get_error_message(wifiErr)); + return false; + } + + return true; +} + +gboolean WiFiManager::_WiFiDeactivate(GMainLoop * mainLoop, gpointer userData) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + wifiErr = wifi_manager_deactivate(sInstance.mWiFiManagerHandle, _DeactivateCb, mainLoop); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi deactivation is requested"); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: request WiFi deactivation [%s]", get_error_message(wifiErr)); + return false; + } + + return true; +} + +gboolean WiFiManager::_WiFiScan(GMainLoop * mainLoop, gpointer userData) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + wifiErr = wifi_manager_scan(sInstance.mWiFiManagerHandle, _ScanFinishedCb, mainLoop); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi scan is requested"); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: request WiFi scan [%s]", get_error_message(wifiErr)); + return false; + } + + return true; +} + +gboolean WiFiManager::_WiFiConnect(GMainLoop * mainLoop, gpointer userData) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + wifi_manager_ap_h ap = static_cast(userData); + + wifiErr = wifi_manager_connect(sInstance.mWiFiManagerHandle, ap, _ConnectedCb, mainLoop); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi connect is requested"); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: request WiFi connect [%s]", get_error_message(wifiErr)); + wifi_manager_ap_destroy(ap); + return false; + } + + wifi_manager_ap_destroy(ap); + return true; +} + +void WiFiManager::_WiFiSetStates(void) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + bool isWifiActivated = false; + + wifiErr = wifi_manager_is_activated(mWiFiManagerHandle, &isWifiActivated); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + mDeviceState = isWifiActivated ? WIFI_MANAGER_DEVICE_STATE_ACTIVATED : WIFI_MANAGER_DEVICE_STATE_DEACTIVATED; + ChipLogProgress(DeviceLayer, "Set WiFi device state [%s]", __WiFiDeviceStateToStr(mDeviceState)); + } + + wifiErr = wifi_manager_get_module_state(mWiFiManagerHandle, &mModuleState); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Set WiFi module state [%s]", __WiFiModuleStateToStr(mModuleState)); + } + + wifiErr = wifi_manager_get_connection_state(mWiFiManagerHandle, &mConnectionState); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Set WiFi connection state [%s]", __WiFiConnectionStateToStr(mConnectionState)); + } +} + +void WiFiManager::_WiFiSetCallbacks(void) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + wifiErr = wifi_manager_set_device_state_changed_cb(mWiFiManagerHandle, _DeviceStateChangedCb, NULL); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Set WiFi device state changed callback"); + } + + wifiErr = wifi_manager_set_module_state_changed_cb(mWiFiManagerHandle, _ModuleStateChangedCb, NULL); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Set WiFi module state changed callback"); + } + + wifiErr = wifi_manager_set_connection_state_changed_cb(mWiFiManagerHandle, _ConnectionStateChangedCb, NULL); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Set WiFi connection state changed callback"); + } + + wifiErr = wifi_manager_set_scan_state_changed_cb(mWiFiManagerHandle, _ScanStateChangedCb, NULL); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Set WiFi scan state changed callback"); + } + + wifiErr = wifi_manager_set_rssi_level_changed_cb(mWiFiManagerHandle, _RssiLevelChangedCb, NULL); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Set WiFi rssi level changed callback"); + } + + wifiErr = wifi_manager_set_background_scan_cb(mWiFiManagerHandle, _BackgroundScanCb, NULL); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Set WiFi background scan callback"); + } + + wifiErr = wifi_manager_set_ip_conflict_cb(mWiFiManagerHandle, _IPConflictCb, NULL); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Set WiFi IP conflict callback"); + } +} + +void WiFiManager::_WiFiUnsetCallbacks(void) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + wifiErr = wifi_manager_unset_device_state_changed_cb(mWiFiManagerHandle); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Unset WiFi device state changed callback"); + } + + wifiErr = wifi_manager_unset_module_state_changed_cb(mWiFiManagerHandle); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Unset WiFi module state changed callback"); + } + + wifiErr = wifi_manager_unset_connection_state_changed_cb(mWiFiManagerHandle); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Unset WiFi connection state changed callback"); + } + + wifiErr = wifi_manager_unset_scan_state_changed_cb(mWiFiManagerHandle); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Unset WiFi scan state changed callback"); + } + + wifiErr = wifi_manager_unset_rssi_level_changed_cb(mWiFiManagerHandle); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Unset WiFi rssi level changed callback"); + } + + wifiErr = wifi_manager_unset_background_scan_cb(mWiFiManagerHandle); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Unset WiFi background scan callback"); + } + + wifiErr = wifi_manager_unset_ip_conflict_cb(mWiFiManagerHandle); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Unset WiFi IP conflict callback"); + } +} + +void WiFiManager::_WiFiSetDeviceState(wifi_manager_device_state_e deviceState) +{ + mDeviceState = deviceState; + ChipLogProgress(DeviceLayer, "Set WiFi device state [%s]", __WiFiDeviceStateToStr(mDeviceState)); +} + +void WiFiManager::_WiFiSetModuleState(wifi_manager_module_state_e moduleState) +{ + mModuleState = moduleState; + ChipLogProgress(DeviceLayer, "Set WiFi module state [%s]", __WiFiModuleStateToStr(mModuleState)); +} + +void WiFiManager::_WiFiSetConnectionState(wifi_manager_connection_state_e connectionState) +{ + mConnectionState = connectionState; + ChipLogProgress(DeviceLayer, "Set WiFi connection state [%s]", __WiFiConnectionStateToStr(mConnectionState)); +} + +wifi_manager_ap_h WiFiManager::_WiFiGetFoundAP(void) +{ + int wifiErr = WIFI_MANAGER_ERROR_NONE; + wifi_manager_ap_h foundAp = NULL; + + wifiErr = wifi_manager_foreach_found_ap(mWiFiManagerHandle, _FoundAPCb, &foundAp); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Get found AP list finished"); + } + else + { + ChipLogProgress(DeviceLayer, "FAIL: get found AP list [%s]", get_error_message(wifiErr)); + } + + return foundAp; +} + +void WiFiManager::Init(void) +{ + sInstance.mDeviceState = WIFI_MANAGER_DEVICE_STATE_DEACTIVATED; + sInstance.mModuleState = WIFI_MANAGER_MODULE_STATE_DETACHED; + sInstance.mConnectionState = WIFI_MANAGER_CONNECTION_STATE_DISCONNECTED; + + MainLoop::Instance().Init(_WiFiInitialize); +} + +void WiFiManager::Deinit(void) +{ + sInstance._WiFiDeinitialize(); + MainLoop::Instance().Deinit(); +} + +CHIP_ERROR WiFiManager::IsActivated(bool * isWifiActivated) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + wifiErr = wifi_manager_is_activated(sInstance.mWiFiManagerHandle, isWifiActivated); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi is %s", *isWifiActivated ? "activated" : "deactivated"); + } + else + { + err = CHIP_ERROR_INCORRECT_STATE; + ChipLogProgress(DeviceLayer, "FAIL: check whether WiFi is activated [%s]", get_error_message(wifiErr)); + } + + return err; +} + +CHIP_ERROR WiFiManager::Activate(void) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + int wifiErr = WIFI_MANAGER_ERROR_NONE; + bool isWifiActivated = false; + bool dbusAsyncErr = false; + + wifiErr = wifi_manager_is_activated(sInstance.mWiFiManagerHandle, &isWifiActivated); + VerifyOrExit(wifiErr == WIFI_MANAGER_ERROR_NONE, err = CHIP_ERROR_INCORRECT_STATE; + ChipLogProgress(DeviceLayer, "FAIL: check whether WiFi is activated [%s]", get_error_message(wifiErr))); + + VerifyOrExit(isWifiActivated == false, ChipLogProgress(DeviceLayer, "WiFi is already activated")); + + dbusAsyncErr = MainLoop::Instance().AsyncRequest(_WiFiActivate); + if (dbusAsyncErr == false) + { + err = CHIP_ERROR_INCORRECT_STATE; + } + +exit: + return err; +} + +CHIP_ERROR WiFiManager::Deactivate(void) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + int wifiErr = WIFI_MANAGER_ERROR_NONE; + bool isWifiActivated = false; + bool dbusAsyncErr = false; + + wifiErr = wifi_manager_is_activated(sInstance.mWiFiManagerHandle, &isWifiActivated); + VerifyOrExit(wifiErr == WIFI_MANAGER_ERROR_NONE, err = CHIP_ERROR_INCORRECT_STATE; + ChipLogProgress(DeviceLayer, "FAIL: check whether WiFi is activated [%s]", get_error_message(wifiErr))); + + VerifyOrExit(isWifiActivated == true, ChipLogProgress(DeviceLayer, "WiFi is already deactivated")); + + dbusAsyncErr = MainLoop::Instance().AsyncRequest(_WiFiDeactivate); + if (dbusAsyncErr == false) + { + err = CHIP_ERROR_INCORRECT_STATE; + } + +exit: + return err; +} + +CHIP_ERROR WiFiManager::Connect(const char * ssid, const char * key) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + int wifiErr = WIFI_MANAGER_ERROR_NONE; + bool isWifiActivated = false; + bool dbusAsyncErr = false; + wifi_manager_ap_h foundAp = NULL; + + g_strlcpy(sInstance.mWiFiSSID, ssid, kMaxWiFiSSIDLength + 1); + g_strlcpy(sInstance.mWiFiKey, key, kMaxWiFiKeyLength + 1); + + wifiErr = wifi_manager_is_activated(sInstance.mWiFiManagerHandle, &isWifiActivated); + VerifyOrExit(wifiErr == WIFI_MANAGER_ERROR_NONE, err = CHIP_ERROR_INCORRECT_STATE; + ChipLogProgress(DeviceLayer, "FAIL: check whether WiFi is activated [%s]", get_error_message(wifiErr))); + + VerifyOrExit(isWifiActivated == true, ChipLogProgress(DeviceLayer, "WiFi is deactivated")); + + foundAp = sInstance._WiFiGetFoundAP(); + if (foundAp != NULL) + { + dbusAsyncErr = MainLoop::Instance().AsyncRequest(_WiFiConnect, static_cast(foundAp)); + if (dbusAsyncErr == false) + { + err = CHIP_ERROR_INCORRECT_STATE; + } + } + else + { + dbusAsyncErr = MainLoop::Instance().AsyncRequest(_WiFiScan); + if (dbusAsyncErr == false) + { + err = CHIP_ERROR_INCORRECT_STATE; + } + } + +exit: + return err; +} + +CHIP_ERROR WiFiManager::Disconnect(const char * ssid) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + int wifiErr = WIFI_MANAGER_ERROR_NONE; + bool isWifiActivated = false; + wifi_manager_ap_h foundAp = NULL; + + g_strlcpy(sInstance.mWiFiSSID, ssid, kMaxWiFiSSIDLength + 1); + + wifiErr = wifi_manager_is_activated(sInstance.mWiFiManagerHandle, &isWifiActivated); + VerifyOrExit(wifiErr == WIFI_MANAGER_ERROR_NONE, err = CHIP_ERROR_INCORRECT_STATE; + ChipLogProgress(DeviceLayer, "FAIL: check whether WiFi is activated [%s]", get_error_message(wifiErr))); + + VerifyOrExit(isWifiActivated == true, ChipLogProgress(DeviceLayer, "WiFi is deactivated")); + + foundAp = sInstance._WiFiGetFoundAP(); + VerifyOrExit(foundAp != NULL, ); + + wifiErr = wifi_manager_forget_ap(sInstance.mWiFiManagerHandle, foundAp); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "WiFi is disconnected"); + } + else + { + err = CHIP_ERROR_INCORRECT_STATE; + ChipLogProgress(DeviceLayer, "FAIL: disconnect WiFi [%s]", get_error_message(wifiErr)); + } + + wifi_manager_ap_destroy(foundAp); + +exit: + return err; +} + +CHIP_ERROR WiFiManager::RemoveAllConfigs(void) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + wifiErr = wifi_manager_config_foreach_configuration(sInstance.mWiFiManagerHandle, _ConfigListCb, NULL); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Get config list finished"); + } + else + { + err = CHIP_ERROR_INCORRECT_STATE; + ChipLogProgress(DeviceLayer, "FAIL: get config list [%s]", get_error_message(wifiErr)); + } + + return err; +} + +CHIP_ERROR WiFiManager::GetDeviceState(wifi_manager_device_state_e * deviceState) +{ + *deviceState = sInstance.mDeviceState; + ChipLogProgress(DeviceLayer, "Get WiFi device state [%s]", __WiFiDeviceStateToStr(*deviceState)); + + return CHIP_NO_ERROR; +} + +CHIP_ERROR WiFiManager::SetDeviceState(wifi_manager_device_state_e deviceState) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + + VerifyOrExit(sInstance.mDeviceState != deviceState, ); + + if (deviceState == WIFI_MANAGER_DEVICE_STATE_DEACTIVATED) + { + err = Deactivate(); + } + else + { + err = Activate(); + } + +exit: + return err; +} + +CHIP_ERROR WiFiManager::GetModuleState(wifi_manager_module_state_e * moduleState) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + if (sInstance.mModuleState != WIFI_MANAGER_MODULE_STATE_DETACHED) + { + *moduleState = sInstance.mModuleState; + ChipLogProgress(DeviceLayer, "Get WiFi module state [%s]", __WiFiModuleStateToStr(*moduleState)); + } + else + { + wifiErr = wifi_manager_get_module_state(sInstance.mWiFiManagerHandle, moduleState); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Get WiFi module state [%s]", __WiFiModuleStateToStr(*moduleState)); + } + else + { + err = CHIP_ERROR_INCORRECT_STATE; + ChipLogProgress(DeviceLayer, "FAIL: get WiFi module state [%s]", get_error_message(wifiErr)); + } + } + + return err; +} + +CHIP_ERROR WiFiManager::GetConnectionState(wifi_manager_connection_state_e * connectionState) +{ + CHIP_ERROR err = CHIP_NO_ERROR; + int wifiErr = WIFI_MANAGER_ERROR_NONE; + + if (sInstance.mConnectionState != WIFI_MANAGER_CONNECTION_STATE_FAILURE) + { + *connectionState = sInstance.mConnectionState; + ChipLogProgress(DeviceLayer, "Get WiFi connection state [%s]", __WiFiConnectionStateToStr(*connectionState)); + } + else + { + wifiErr = wifi_manager_get_connection_state(sInstance.mWiFiManagerHandle, connectionState); + if (wifiErr == WIFI_MANAGER_ERROR_NONE) + { + ChipLogProgress(DeviceLayer, "Get WiFi connection state [%s]", __WiFiConnectionStateToStr(*connectionState)); + } + else + { + err = CHIP_ERROR_INCORRECT_STATE; + ChipLogProgress(DeviceLayer, "FAIL: get WiFi connection state [%s]", get_error_message(wifiErr)); + } + } + + return err; +} + +} // namespace Internal +} // namespace DeviceLayer +} // namespace chip +#endif // CHIP_DEVICE_CONFIG_ENABLE_WIFI diff --git a/src/platform/Tizen/WiFiManager.h b/src/platform/Tizen/WiFiManager.h new file mode 100644 index 00000000000000..eb46e9b8e4fbd9 --- /dev/null +++ b/src/platform/Tizen/WiFiManager.h @@ -0,0 +1,102 @@ +/* + * + * Copyright (c) 2021 Project CHIP Authors + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#if CHIP_DEVICE_CONFIG_ENABLE_WIFI +#include + +#include +#include + +namespace chip { +namespace DeviceLayer { +namespace Internal { + +class WiFiManager +{ + friend class ConnectivityManagerImpl; + +public: + void Init(void); + void Deinit(void); + + CHIP_ERROR IsActivated(bool * isWifiActivated); + CHIP_ERROR Activate(void); + CHIP_ERROR Deactivate(void); + CHIP_ERROR Connect(const char * ssid, const char * key); + CHIP_ERROR Disconnect(const char * ssid); + CHIP_ERROR RemoveAllConfigs(void); + + CHIP_ERROR GetDeviceState(wifi_manager_device_state_e * deviceState); + CHIP_ERROR SetDeviceState(wifi_manager_device_state_e deviceState); + CHIP_ERROR GetModuleState(wifi_manager_module_state_e * moduleState); + CHIP_ERROR GetConnectionState(wifi_manager_connection_state_e * connectionState); + +private: + static void _DeviceStateChangedCb(wifi_manager_device_state_e deviceState, void * userData); + static void _ModuleStateChangedCb(wifi_manager_module_state_e moduleState, void * userData); + static void _ConnectionStateChangedCb(wifi_manager_connection_state_e connectionState, wifi_manager_ap_h ap, void * userData); + static void _ScanStateChangedCb(wifi_manager_scan_state_e scanState, void * userData); + static void _RssiLevelChangedCb(wifi_manager_rssi_level_e rssiLevel, void * userData); + static void _BackgroundScanCb(wifi_manager_error_e wifiErr, void * userData); + static void _IPConflictCb(char * mac, wifi_manager_ip_conflict_state_e ipConflictState, void * userData); + static void _ActivateCb(wifi_manager_error_e wifiErr, void * userData); + static void _DeactivateCb(wifi_manager_error_e wifiErr, void * userData); + static void _ScanFinishedCb(wifi_manager_error_e wifiErr, void * userData); + static bool _FoundAPCb(wifi_manager_ap_h ap, void * userData); + static void _ConnectedCb(wifi_manager_error_e wifiErr, void * userData); + static bool _ConfigListCb(const wifi_manager_config_h config, void * userData); + + static gboolean _WiFiInitialize(gpointer userData); + static gboolean _WiFiActivate(GMainLoop * mainLoop, gpointer userData); + static gboolean _WiFiDeactivate(GMainLoop * mainLoop, gpointer userData); + static gboolean _WiFiScan(GMainLoop * mainLoop, gpointer userData); + static gboolean _WiFiConnect(GMainLoop * mainLoop, gpointer userData); + + void _WiFiDeinitialize(void); + void _WiFiSetStates(void); + void _WiFiSetCallbacks(void); + void _WiFiUnsetCallbacks(void); + void _WiFiSetDeviceState(wifi_manager_device_state_e deviceState); + void _WiFiSetModuleState(wifi_manager_module_state_e moduleState); + void _WiFiSetConnectionState(wifi_manager_connection_state_e connectionState); + wifi_manager_ap_h _WiFiGetFoundAP(void); + + friend WiFiManager & WiFiMgr(void); + + static WiFiManager sInstance; + + wifi_manager_h mWiFiManagerHandle; + wifi_manager_device_state_e mDeviceState; + wifi_manager_module_state_e mModuleState; + wifi_manager_connection_state_e mConnectionState; + + char mWiFiSSID[kMaxWiFiSSIDLength + 1]; + char mWiFiKey[kMaxWiFiKeyLength + 1]; +}; + +inline WiFiManager & WiFiMgr() +{ + return WiFiManager::sInstance; +} + +} // namespace Internal +} // namespace DeviceLayer +} // namespace chip + +#endif // CHIP_DEVICE_CONFIG_ENABLE_WIFI