/* -*- Mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 40 -*- */ /* vim: set ts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this file, * You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "base/basictypes.h" #include "BluetoothManager.h" #include "BluetoothCommon.h" #include "BluetoothAdapter.h" #include "BluetoothService.h" #include "BluetoothReplyRunnable.h" #include "nsContentUtils.h" #include "nsDOMClassInfo.h" #include "nsDOMEvent.h" #include "nsIDOMDOMRequest.h" #include "nsIPermissionManager.h" #include "nsThreadUtils.h" #include "nsXPCOMCIDInternal.h" #include "mozilla/Util.h" #include "mozilla/dom/bluetooth/BluetoothTypes.h" using namespace mozilla; USING_BLUETOOTH_NAMESPACE DOMCI_DATA(BluetoothManager, BluetoothManager) NS_IMPL_CYCLE_COLLECTION_CLASS(BluetoothManager) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(BluetoothManager, nsDOMEventTargetHelper) NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(BluetoothManager, nsDOMEventTargetHelper) NS_IMPL_CYCLE_COLLECTION_UNLINK_END NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(BluetoothManager) NS_INTERFACE_MAP_ENTRY(nsIDOMBluetoothManager) NS_DOM_INTERFACE_MAP_ENTRY_CLASSINFO(BluetoothManager) NS_INTERFACE_MAP_END_INHERITING(nsDOMEventTargetHelper) NS_IMPL_ADDREF_INHERITED(BluetoothManager, nsDOMEventTargetHelper) NS_IMPL_RELEASE_INHERITED(BluetoothManager, nsDOMEventTargetHelper) class GetAdapterTask : public BluetoothReplyRunnable { public: GetAdapterTask(BluetoothManager* aManager, nsIDOMDOMRequest* aReq) : BluetoothReplyRunnable(aReq), mManagerPtr(aManager) { } bool ParseSuccessfulReply(jsval* aValue) { nsCOMPtr adapter; *aValue = JSVAL_VOID; const InfallibleTArray& v = mReply->get_BluetoothReplySuccess().value().get_ArrayOfBluetoothNamedValue(); adapter = BluetoothAdapter::Create(mManagerPtr->GetOwner(), v); nsresult rv; nsIScriptContext* sc = mManagerPtr->GetContextForEventHandlers(&rv); if (!sc) { NS_WARNING("Cannot create script context!"); SetError(NS_LITERAL_STRING("BluetoothScriptContextError")); return false; } rv = nsContentUtils::WrapNative(sc->GetNativeContext(), sc->GetNativeGlobal(), adapter, aValue); bool result = NS_SUCCEEDED(rv); if (!result) { NS_WARNING("Cannot create native object!"); SetError(NS_LITERAL_STRING("BluetoothNativeObjectError")); } return result; } void ReleaseMembers() { BluetoothReplyRunnable::ReleaseMembers(); mManagerPtr = nullptr; } private: nsRefPtr mManagerPtr; }; class ToggleBtResultTask : public nsRunnable { public: ToggleBtResultTask(BluetoothManager* aManager, bool aEnabled) : mManagerPtr(aManager), mEnabled(aEnabled) { } NS_IMETHOD Run() { MOZ_ASSERT(NS_IsMainThread()); mManagerPtr->FireEnabledDisabledEvent(mEnabled); // mManagerPtr must be null before returning to prevent the background // thread from racing to release it during the destruction of this runnable. mManagerPtr = nullptr; return NS_OK; } private: nsRefPtr mManagerPtr; bool mEnabled; }; nsresult BluetoothManager::FireEnabledDisabledEvent(bool aEnabled) { return DispatchTrustedEvent(aEnabled ? NS_LITERAL_STRING("enabled") : NS_LITERAL_STRING("disabled")); } BluetoothManager::BluetoothManager(nsPIDOMWindow *aWindow) : BluetoothPropertyContainer(BluetoothObjectType::TYPE_MANAGER) { MOZ_ASSERT(aWindow); BindToOwner(aWindow); mPath.AssignLiteral("/"); } BluetoothManager::~BluetoothManager() { BluetoothService* bs = BluetoothService::Get(); if (bs) { bs->UnregisterManager(this); } } void BluetoothManager::SetPropertyByValue(const BluetoothNamedValue& aValue) { #ifdef DEBUG const nsString& name = aValue.name(); nsCString warningMsg; warningMsg.AssignLiteral("Not handling manager property: "); warningMsg.Append(NS_ConvertUTF16toUTF8(name)); NS_WARNING(warningMsg.get()); #endif } NS_IMETHODIMP BluetoothManager::GetEnabled(bool* aEnabled) { BluetoothService* bs = BluetoothService::Get(); if (!bs) { NS_WARNING("BluetoothService not available!"); return NS_ERROR_FAILURE; } *aEnabled = bs->IsEnabled(); return NS_OK; } NS_IMETHODIMP BluetoothManager::GetDefaultAdapter(nsIDOMDOMRequest** aAdapter) { BluetoothService* bs = BluetoothService::Get(); if (!bs) { NS_WARNING("BluetoothService not available!"); return NS_ERROR_FAILURE; } nsCOMPtr rs = do_GetService("@mozilla.org/dom/dom-request-service;1"); if (!rs) { NS_WARNING("No DOMRequest Service!"); return NS_ERROR_FAILURE; } nsCOMPtr request; nsresult rv = rs->CreateRequest(GetOwner(), getter_AddRefs(request)); if (NS_FAILED(rv)) { NS_WARNING("Can't create DOMRequest!"); return NS_ERROR_FAILURE; } nsRefPtr results = new GetAdapterTask(this, request); if (NS_FAILED(bs->GetDefaultAdapterPathInternal(results))) { return NS_ERROR_FAILURE; } request.forget(aAdapter); return NS_OK; } // static already_AddRefed BluetoothManager::Create(nsPIDOMWindow* aWindow) { MOZ_ASSERT(NS_IsMainThread()); MOZ_ASSERT(aWindow); BluetoothService* bs = BluetoothService::Get(); if (!bs) { NS_WARNING("BluetoothService not available!"); return nullptr; } nsRefPtr manager = new BluetoothManager(aWindow); bs->RegisterManager(manager); return manager.forget(); } nsresult NS_NewBluetoothManager(nsPIDOMWindow* aWindow, nsIDOMBluetoothManager** aBluetoothManager) { NS_ASSERTION(aWindow, "Null pointer!"); nsCOMPtr permMgr = do_GetService(NS_PERMISSIONMANAGER_CONTRACTID); NS_ENSURE_TRUE(permMgr, NS_ERROR_UNEXPECTED); uint32_t permission; nsresult rv = permMgr->TestPermissionFromWindow(aWindow, "mozBluetooth", &permission); NS_ENSURE_SUCCESS(rv, rv); nsRefPtr bluetoothManager; if (permission == nsIPermissionManager::ALLOW_ACTION) { bluetoothManager = BluetoothManager::Create(aWindow); } bluetoothManager.forget(aBluetoothManager); return NS_OK; } void BluetoothManager::Notify(const BluetoothSignal& aData) { if (aData.name().EqualsLiteral("AdapterAdded")) { DispatchTrustedEvent(NS_LITERAL_STRING("adapteradded")); } else { #ifdef DEBUG nsCString warningMsg; warningMsg.AssignLiteral("Not handling manager signal: "); warningMsg.Append(NS_ConvertUTF16toUTF8(aData.name())); NS_WARNING(warningMsg.get()); #endif } } NS_IMETHODIMP BluetoothManager::IsConnected(uint16_t aProfileId, bool* aConnected) { BluetoothService* bs = BluetoothService::Get(); if (!bs) { NS_WARNING("BluetoothService not available!"); return NS_ERROR_FAILURE; } *aConnected = bs->IsConnected(aProfileId); return NS_OK; } NS_IMPL_EVENT_HANDLER(BluetoothManager, enabled) NS_IMPL_EVENT_HANDLER(BluetoothManager, disabled) NS_IMPL_EVENT_HANDLER(BluetoothManager, adapteradded)