• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2023 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "napi_bluetooth_access.h"
17 
18 #include "bluetooth_log.h"
19 #include "bluetooth_errorcode.h"
20 #include "bluetooth_host.h"
21 
22 #include "napi_bluetooth_error.h"
23 #include "napi_bluetooth_access_observer.h"
24 #include "napi_bluetooth_utils.h"
25 #include "napi_bluetooth_spp_client.h"
26 #include "../parser/napi_parser_utils.h"
27 
28 namespace OHOS {
29 namespace Bluetooth {
30 namespace {
31 NapiBluetoothAccessObserver g_bluetoothAccessObserver;
32 }  // namespace
33 
DefineAccessJSFunction(napi_env env,napi_value exports)34 napi_value NapiAccess::DefineAccessJSFunction(napi_env env, napi_value exports)
35 {
36     HILOGI("start");
37     RegisterAccessObserverToHost();
38     AccessPropertyValueInit(env, exports);
39     napi_property_descriptor desc[] = {
40         DECLARE_NAPI_FUNCTION("getState", GetState),
41         DECLARE_NAPI_FUNCTION("enableBluetooth", EnableBluetooth),
42         DECLARE_NAPI_FUNCTION("disableBluetooth", DisableBluetooth),
43 #ifdef BLUETOOTH_API_SINCE_10
44         DECLARE_NAPI_FUNCTION("on", RegisterAccessObserver),
45         DECLARE_NAPI_FUNCTION("off", DeregisterAccessObserver),
46 #endif
47     };
48     napi_define_properties(env, exports, sizeof(desc) / sizeof(desc[0]), desc);
49     HILOGI("end");
50     return exports;
51 }
52 
RegisterAccessObserverToHost()53 void NapiAccess::RegisterAccessObserverToHost()
54 {
55     HILOGI("start");
56     BluetoothHost *host = &BluetoothHost::GetDefaultHost();
57     host->RegisterObserver(g_bluetoothAccessObserver);
58 }
59 
EnableBluetooth(napi_env env,napi_callback_info info)60 napi_value NapiAccess::EnableBluetooth(napi_env env, napi_callback_info info)
61 {
62     HILOGI("start");
63     BluetoothHost *host = &BluetoothHost::GetDefaultHost();
64     int32_t ret = host->EnableBle();
65     NAPI_BT_ASSERT_RETURN_FALSE(env, ret == BT_NO_ERROR, ret);
66     if (ret == BT_NO_ERROR) {
67         SetCurrentAppOperate(true);
68     }
69     return NapiGetBooleanTrue(env);
70 }
71 
DisableBluetooth(napi_env env,napi_callback_info info)72 napi_value NapiAccess::DisableBluetooth(napi_env env, napi_callback_info info)
73 {
74     HILOGI("start");
75     BluetoothHost *host = &BluetoothHost::GetDefaultHost();
76     // only ble
77     int state = BTStateID::STATE_TURN_OFF;
78     int ret = host->GetBtState(state);
79     NAPI_BT_ASSERT_RETURN_FALSE(env, ret == BT_NO_ERROR, ret);
80     if (state == BTStateID::STATE_TURN_OFF) {
81         ret = host->DisableBle();
82         NAPI_BT_ASSERT_RETURN_FALSE(env, ret == BT_NO_ERROR, ret);
83     } else {
84         ret = host->DisableBt();
85         NAPI_BT_ASSERT_RETURN_FALSE(env, ret == BT_NO_ERROR, ret);
86     }
87 
88     if (ret == BT_NO_ERROR) {
89         SetCurrentAppOperate(true);
90     }
91     return NapiGetBooleanTrue(env);
92 }
93 
GetState(napi_env env,napi_callback_info info)94 napi_value NapiAccess::GetState(napi_env env, napi_callback_info info)
95 {
96     HILOGI("enter");
97     BluetoothHost *host = &BluetoothHost::GetDefaultHost();
98     int32_t state = BTStateID::STATE_TURN_OFF;
99     int32_t err = host->GetBtState(state);
100     NAPI_BT_ASSERT_RETURN_FALSE(env, err == NO_ERROR, err);
101     int32_t status = static_cast<int32_t>(BluetoothState::STATE_OFF);
102     switch (state) {
103         case BTStateID::STATE_TURNING_ON:
104             HILOGI("STATE_TURNING_ON(1)");
105             status = static_cast<int32_t>(BluetoothState::STATE_TURNING_ON);
106             break;
107         case BTStateID::STATE_TURN_ON:
108             HILOGI("STATE_ON(2)");
109             status = static_cast<int32_t>(BluetoothState::STATE_ON);
110             break;
111         case BTStateID::STATE_TURNING_OFF:
112             HILOGI("STATE_TURNING_OFF(3)");
113             status = static_cast<int32_t>(BluetoothState::STATE_TURNING_OFF);
114             break;
115         case BTStateID::STATE_TURN_OFF:
116             HILOGI("STATE_OFF(0)");
117             status = static_cast<int32_t>(BluetoothState::STATE_OFF);
118             break;
119         default:
120             HILOGE("get state failed");
121             break;
122     }
123 
124     bool enableBle = host->IsBleEnabled();
125     if (enableBle && (state == BTStateID::STATE_TURN_OFF)) {
126         HILOGI("BR off and BLE on, STATE_BLE_ON(5)");
127         status = static_cast<int32_t>(BluetoothState::STATE_BLE_ON);
128     } else if (!enableBle && (state == BTStateID::STATE_TURN_OFF)) {
129         status = static_cast<int32_t>(BluetoothState::STATE_OFF);
130     }
131 
132     napi_value result = nullptr;
133     napi_create_int32(env, status, &result);
134     return result;
135 }
136 
AccessPropertyValueInit(napi_env env,napi_value exports)137 napi_value NapiAccess::AccessPropertyValueInit(napi_env env, napi_value exports)
138 {
139     HILOGI("start");
140     napi_value stateObj = StateChangeInit(env);
141     napi_property_descriptor exportFuncs[] = {
142         DECLARE_NAPI_PROPERTY("BluetoothState", stateObj),
143     };
144     napi_define_properties(env, exports, sizeof(exportFuncs) / sizeof(*exportFuncs), exportFuncs);
145     HILOGI("end");
146     return exports;
147 }
148 
StateChangeInit(napi_env env)149 napi_value NapiAccess::StateChangeInit(napi_env env)
150 {
151     HILOGI("enter");
152     napi_value state = nullptr;
153     napi_create_object(env, &state);
154     SetNamedPropertyByInteger(env, state, static_cast<int>(BluetoothState::STATE_OFF), "STATE_OFF");
155     SetNamedPropertyByInteger(env, state, static_cast<int>(BluetoothState::STATE_TURNING_ON), "STATE_TURNING_ON");
156     SetNamedPropertyByInteger(env, state, static_cast<int>(BluetoothState::STATE_ON), "STATE_ON");
157     SetNamedPropertyByInteger(env, state, static_cast<int>(BluetoothState::STATE_TURNING_OFF), "STATE_TURNING_OFF");
158     SetNamedPropertyByInteger(
159         env, state, static_cast<int>(BluetoothState::STATE_BLE_TURNING_ON), "STATE_BLE_TURNING_ON");
160     SetNamedPropertyByInteger(env, state, static_cast<int>(BluetoothState::STATE_BLE_ON), "STATE_BLE_ON");
161     SetNamedPropertyByInteger(
162         env, state, static_cast<int>(BluetoothState::STATE_BLE_TURNING_OFF), "STATE_BLE_TURNING_OFF");
163     return state;
164 }
165 
IsValidObserverType(const std::string & outType)166 static bool IsValidObserverType(const std::string &outType)
167 {
168     return outType == REGISTER_STATE_CHANGE_TYPE;
169 }
170 
NapiParseObserverType(napi_env env,napi_value value,std::string & outType)171 static napi_status NapiParseObserverType(napi_env env, napi_value value, std::string &outType)
172 {
173     std::string type{};
174     NAPI_BT_CALL_RETURN(NapiParseString(env, value, type));
175     NAPI_BT_RETURN_IF(!IsValidObserverType(type), "Invalid type", napi_invalid_arg);
176     outType = std::move(type);
177     return napi_ok;
178 }
179 
CheckAccessObserverParams(napi_env env,size_t argc,napi_value * argv,std::string & outType,std::shared_ptr<BluetoothCallbackInfo> & outpCallbackInfo)180 static napi_status CheckAccessObserverParams(napi_env env, size_t argc, napi_value *argv, std::string &outType,
181     std::shared_ptr<BluetoothCallbackInfo> &outpCallbackInfo)
182 {
183     NAPI_BT_RETURN_IF(argc != ARGS_SIZE_TWO, "Requires 2 arguments.", napi_invalid_arg);
184     std::string type;
185     NAPI_BT_CALL_RETURN(NapiParseObserverType(env, argv[PARAM0], type));
186     napi_valuetype valueType = napi_undefined;
187     NAPI_BT_CALL_RETURN(napi_typeof(env, argv[PARAM1], &valueType));
188     NAPI_BT_RETURN_IF(valueType != napi_function, "Requires a function argument", napi_function_expected);
189     NAPI_BT_CALL_RETURN(napi_create_reference(env, argv[PARAM1], 1, &outpCallbackInfo->callback_));
190 
191     outType = type;
192     outpCallbackInfo->env_ = env;
193     return napi_ok;
194 }
195 
RegisterAccessObserver(napi_env env,napi_callback_info info)196 napi_value NapiAccess::RegisterAccessObserver(napi_env env, napi_callback_info info)
197 {
198     // fsafasfasf
199     HILOGI("enter");
200     size_t argc = ARGS_SIZE_TWO;
201     napi_value argv[ARGS_SIZE_TWO] = {0};
202     napi_get_cb_info(env, info, &argc, argv, nullptr, nullptr);
203     HILOGI("argc: %{public}zu", argc);
204     std::string type;
205     std::shared_ptr<BluetoothCallbackInfo> pCallbackInfo = std::make_shared<BluetoothCallbackInfo>();
206     auto status = CheckAccessObserverParams(env, argc, argv, type, pCallbackInfo);
207     NAPI_BT_ASSERT_RETURN_UNDEF(env, status == napi_ok, BT_ERR_INVALID_PARAM);
208     g_bluetoothAccessObserver.stateChangeCallback = pCallbackInfo;
209     HILOGI("%{public}s is registered", type.c_str());
210     napi_value ret = nullptr;
211     napi_get_undefined(env, &ret);
212     return ret;
213 }
214 
CheckAccessDeregisterObserver(napi_env env,napi_callback_info info)215 static napi_status CheckAccessDeregisterObserver(napi_env env, napi_callback_info info)
216 {
217     size_t argc = ARGS_SIZE_TWO;
218     napi_value argv[ARGS_SIZE_TWO] = {0};
219     napi_value thisVar = nullptr;
220     NAPI_BT_CALL_RETURN(napi_get_cb_info(env, info, &argc, argv, &thisVar, nullptr));
221     if (argc == ARGS_SIZE_ONE) {
222         std::string type;
223         NAPI_BT_CALL_RETURN(NapiParseObserverType(env, argv[PARAM0], type));
224         g_bluetoothAccessObserver.stateChangeCallback = nullptr;
225     } else {
226         NAPI_BT_RETURN_IF(argc != ARGS_SIZE_TWO, "Requires 2 arguments.", napi_invalid_arg);
227         std::string type;
228         NAPI_BT_CALL_RETURN(NapiParseObserverType(env, argv[PARAM0], type));
229         std::shared_ptr<BluetoothCallbackInfo> pCallbackInfo = std::make_shared<BluetoothCallbackInfo>();
230         NAPI_BT_RETURN_IF(pCallbackInfo == nullptr, "pCallbackInfo is nullptr.", napi_invalid_arg);
231         pCallbackInfo->env_ = env;
232         NAPI_BT_CALL_RETURN(NapiIsFunction(env, argv[PARAM1]));
233         NAPI_BT_CALL_RETURN(napi_create_reference(env, argv[PARAM1], 1, &pCallbackInfo->callback_));
234         napi_value callback = 0;
235         napi_value undefined = 0;
236         napi_value callResult = 0;
237         napi_get_undefined(pCallbackInfo->env_, &undefined);
238         napi_value result = nullptr;
239         napi_create_int32(pCallbackInfo->env_, static_cast<int32_t>(BluetoothState::STATE_OFF), &result);
240         napi_get_reference_value(pCallbackInfo->env_, pCallbackInfo->callback_, &callback);
241         napi_call_function(pCallbackInfo->env_, undefined, callback, ARGS_SIZE_ONE, &result, &callResult);
242         g_bluetoothAccessObserver.stateChangeCallback = nullptr;
243     }
244     return napi_ok;
245 }
246 
DeregisterAccessObserver(napi_env env,napi_callback_info info)247 napi_value NapiAccess::DeregisterAccessObserver(napi_env env, napi_callback_info info)
248 {
249     HILOGI("enter");
250     auto status = CheckAccessDeregisterObserver(env, info);
251     NAPI_BT_ASSERT_RETURN_UNDEF(env, status == napi_ok, BT_ERR_INVALID_PARAM);
252     return NapiGetUndefinedRet(env);
253 }
254 }  // namespace Bluetooth
255 }  // namespace OHOS
256