mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-08 20:47:44 +00:00
294 lines
6.8 KiB
C++
294 lines
6.8 KiB
C++
/* -*- 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 "BluetoothScoManager.h"
|
|
|
|
#include "BluetoothReplyRunnable.h"
|
|
#include "BluetoothService.h"
|
|
#include "BluetoothSocket.h"
|
|
#include "BluetoothUtils.h"
|
|
|
|
#include "mozilla/Services.h"
|
|
#include "mozilla/StaticPtr.h"
|
|
#include "mozilla/dom/bluetooth/BluetoothTypes.h"
|
|
#include "nsContentUtils.h"
|
|
#include "nsIAudioManager.h"
|
|
#include "nsIObserverService.h"
|
|
|
|
#define BLUETOOTH_SCO_STATUS_CHANGED "bluetooth-sco-status-changed"
|
|
|
|
using namespace mozilla;
|
|
using namespace mozilla::ipc;
|
|
USING_BLUETOOTH_NAMESPACE
|
|
|
|
class mozilla::dom::bluetooth::BluetoothScoManagerObserver : public nsIObserver
|
|
{
|
|
public:
|
|
NS_DECL_ISUPPORTS
|
|
NS_DECL_NSIOBSERVER
|
|
|
|
BluetoothScoManagerObserver()
|
|
{
|
|
}
|
|
|
|
bool Init()
|
|
{
|
|
nsCOMPtr<nsIObserverService> obs = services::GetObserverService();
|
|
MOZ_ASSERT(obs);
|
|
|
|
if (NS_FAILED(obs->AddObserver(this, NS_XPCOM_SHUTDOWN_OBSERVER_ID, false))) {
|
|
NS_WARNING("Failed to add shutdown observer!");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool Shutdown()
|
|
{
|
|
nsCOMPtr<nsIObserverService> obs = services::GetObserverService();
|
|
if (!obs ||
|
|
(NS_FAILED(obs->RemoveObserver(this, NS_XPCOM_SHUTDOWN_OBSERVER_ID)))) {
|
|
NS_WARNING("Can't unregister observers!");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
~BluetoothScoManagerObserver()
|
|
{
|
|
Shutdown();
|
|
}
|
|
};
|
|
|
|
void
|
|
BluetoothScoManager::NotifyAudioManager(const nsAString& aAddress)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
nsCOMPtr<nsIObserverService> obs =
|
|
do_GetService("@mozilla.org/observer-service;1");
|
|
NS_ENSURE_TRUE_VOID(obs);
|
|
|
|
const PRUnichar* addr =
|
|
aAddress.IsEmpty() ? nullptr : aAddress.BeginReading();
|
|
|
|
if (NS_FAILED(obs->NotifyObservers(nullptr,
|
|
BLUETOOTH_SCO_STATUS_CHANGED,
|
|
addr))) {
|
|
NS_WARNING("Failed to notify bluetooth-sco-status-changed observsers!");
|
|
return;
|
|
}
|
|
}
|
|
|
|
NS_IMPL_ISUPPORTS1(BluetoothScoManagerObserver, nsIObserver)
|
|
|
|
namespace {
|
|
StaticAutoPtr<BluetoothScoManager> gBluetoothScoManager;
|
|
StaticRefPtr<BluetoothScoManagerObserver> sScoObserver;
|
|
bool gInShutdown = false;
|
|
} // anonymous namespace
|
|
|
|
NS_IMETHODIMP
|
|
BluetoothScoManagerObserver::Observe(nsISupports* aSubject,
|
|
const char* aTopic,
|
|
const PRUnichar* aData)
|
|
{
|
|
MOZ_ASSERT(gBluetoothScoManager);
|
|
if (!strcmp(aTopic, NS_XPCOM_SHUTDOWN_OBSERVER_ID)) {
|
|
return gBluetoothScoManager->HandleShutdown();
|
|
}
|
|
|
|
MOZ_ASSERT(false, "BluetoothScoManager got unexpected topic!");
|
|
return NS_ERROR_UNEXPECTED;
|
|
}
|
|
|
|
BluetoothScoManager::BluetoothScoManager()
|
|
{
|
|
}
|
|
|
|
bool
|
|
BluetoothScoManager::Init()
|
|
{
|
|
mSocket = new BluetoothSocket(this,
|
|
BluetoothSocketType::SCO,
|
|
true,
|
|
false);
|
|
mPrevSocketStatus = mSocket->GetConnectionStatus();
|
|
|
|
sScoObserver = new BluetoothScoManagerObserver();
|
|
if (!sScoObserver->Init()) {
|
|
NS_WARNING("Cannot set up SCO observers!");
|
|
}
|
|
return true;
|
|
}
|
|
|
|
BluetoothScoManager::~BluetoothScoManager()
|
|
{
|
|
Cleanup();
|
|
}
|
|
|
|
void
|
|
BluetoothScoManager::Cleanup()
|
|
{
|
|
sScoObserver->Shutdown();
|
|
sScoObserver = nullptr;
|
|
}
|
|
|
|
//static
|
|
BluetoothScoManager*
|
|
BluetoothScoManager::Get()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
// If we already exist, exit early
|
|
if (gBluetoothScoManager) {
|
|
return gBluetoothScoManager;
|
|
}
|
|
|
|
// If we're in shutdown, don't create a new instance
|
|
if (gInShutdown) {
|
|
NS_WARNING("BluetoothScoManager can't be created during shutdown");
|
|
return nullptr;
|
|
}
|
|
|
|
// Create new instance, register, return
|
|
BluetoothScoManager* manager = new BluetoothScoManager();
|
|
NS_ENSURE_TRUE(manager->Init(), nullptr);
|
|
|
|
gBluetoothScoManager = manager;
|
|
return gBluetoothScoManager;
|
|
}
|
|
|
|
// Virtual function of class SocketConsumer
|
|
void
|
|
BluetoothScoManager::ReceiveSocketData(BluetoothSocket* aSocket,
|
|
nsAutoPtr<UnixSocketRawData>& aMessage)
|
|
{
|
|
// SCO socket do nothing here
|
|
MOZ_NOT_REACHED("This should never be called!");
|
|
}
|
|
|
|
nsresult
|
|
BluetoothScoManager::HandleShutdown()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
gInShutdown = true;
|
|
mSocket->Disconnect();
|
|
gBluetoothScoManager = nullptr;
|
|
return NS_OK;
|
|
}
|
|
|
|
bool
|
|
BluetoothScoManager::Connect(const nsAString& aDeviceAddress)
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (gInShutdown) {
|
|
MOZ_ASSERT(false, "Connect called while in shutdown!");
|
|
return false;
|
|
}
|
|
|
|
SocketConnectionStatus status = mSocket->GetConnectionStatus();
|
|
if (status == SocketConnectionStatus::SOCKET_CONNECTED ||
|
|
status == SocketConnectionStatus::SOCKET_CONNECTING) {
|
|
NS_WARNING("SCO connection exists or is being established");
|
|
return false;
|
|
}
|
|
|
|
mSocket->Disconnect();
|
|
|
|
BluetoothService* bs = BluetoothService::Get();
|
|
NS_ENSURE_TRUE(bs, false);
|
|
|
|
nsresult rv = bs->GetScoSocket(aDeviceAddress,
|
|
true,
|
|
false,
|
|
mSocket);
|
|
|
|
return NS_FAILED(rv) ? false : true;
|
|
}
|
|
|
|
bool
|
|
BluetoothScoManager::Listen()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (gInShutdown) {
|
|
MOZ_ASSERT(false, "Listen called while in shutdown!");
|
|
return false;
|
|
}
|
|
|
|
if (mSocket->GetConnectionStatus() ==
|
|
SocketConnectionStatus::SOCKET_LISTENING) {
|
|
NS_WARNING("BluetoothScoManager has been already listening");
|
|
return true;
|
|
}
|
|
|
|
mSocket->Disconnect();
|
|
|
|
if (!mSocket->Listen(-1)) {
|
|
NS_WARNING("[SCO] Can't listen on socket!");
|
|
return false;
|
|
}
|
|
|
|
mPrevSocketStatus = mSocket->GetConnectionStatus();
|
|
return true;
|
|
}
|
|
|
|
void
|
|
BluetoothScoManager::Disconnect()
|
|
{
|
|
mSocket->Disconnect();
|
|
}
|
|
|
|
void
|
|
BluetoothScoManager::OnConnectSuccess(BluetoothSocket* aSocket)
|
|
{
|
|
MOZ_ASSERT(aSocket == mSocket);
|
|
|
|
nsString address;
|
|
mSocket->GetAddress(address);
|
|
NotifyAudioManager(address);
|
|
|
|
mPrevSocketStatus = mSocket->GetConnectionStatus();
|
|
}
|
|
|
|
void
|
|
BluetoothScoManager::OnConnectError(BluetoothSocket* aSocket)
|
|
{
|
|
MOZ_ASSERT(aSocket == mSocket);
|
|
|
|
mSocket->Disconnect();
|
|
mPrevSocketStatus = mSocket->GetConnectionStatus();
|
|
Listen();
|
|
}
|
|
|
|
void
|
|
BluetoothScoManager::OnDisconnect(BluetoothSocket* aSocket)
|
|
{
|
|
MOZ_ASSERT(aSocket == mSocket);
|
|
|
|
if (mPrevSocketStatus == SocketConnectionStatus::SOCKET_CONNECTED) {
|
|
Listen();
|
|
|
|
nsString address = NS_LITERAL_STRING("");
|
|
NotifyAudioManager(address);
|
|
}
|
|
}
|
|
|
|
bool
|
|
BluetoothScoManager::IsConnected()
|
|
{
|
|
if (mSocket) {
|
|
return mSocket->GetConnectionStatus() ==
|
|
SocketConnectionStatus::SOCKET_CONNECTED;
|
|
}
|
|
|
|
return false;
|
|
}
|