The typedef for DBusGObjectPath is only present in newer versions of dbus-glib. It is equivalent to a gchar. Change-Id: I1660234760fde7c7d88721f39d86a820c89040cc
173 lines
5.2 KiB
C++
173 lines
5.2 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*
|
|
* This file is part of the LibreOffice project.
|
|
*
|
|
* 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 "BluetoothServer.hxx"
|
|
#include <stdio.h>
|
|
|
|
#if defined(LINUX) && defined(ENABLE_DBUS)
|
|
#include <glib.h>
|
|
#include <dbus/dbus-glib.h>
|
|
#include <sys/unistd.h>
|
|
#include <sys/socket.h>
|
|
#include "bluetooth/bluetooth.h"
|
|
#include "bluetooth/rfcomm.h"
|
|
#endif
|
|
|
|
// FIXME: move this into an external file and look at sharing definitions
|
|
// across OS's (i.e. UUID and port ).
|
|
// Also look at determining which ports are available.
|
|
#define BLUETOOTH_SERVICE_RECORD "<?xml version='1.0' encoding= 'UTF-8' ?><record><attribute id='0x0001'><sequence><uuid value='0x1101' /></sequence></attribute><attribute id='0x0004'><sequence><sequence><uuid value='0x0100' /></sequence><sequence><uuid value='0x0003' /><uint8 value='0x05' /></sequence></sequence></attribute><attribute id='0x0005'><sequence><uuid value='0x1002' /></sequence></attribute><attribute id='0x0006'><sequence><uint16 value='0x656e' /><uint16 value='0x006a' /><uint16 value='0x0100' /></sequence></attribute><attribute id='0x0009'><sequence><sequence><uuid value='0x1101' /><uint16 value='0x0100' /></sequence></sequence></attribute><attribute id='0x0100'><text value='Serial Port' /></attribute><attribute id='0x0101'><text value='COM Port' /></attribute></record>"
|
|
|
|
#include "Communicator.hxx"
|
|
|
|
using namespace sd;
|
|
|
|
BluetoothServer::BluetoothServer( std::vector<Communicator*>* pCommunicators ):
|
|
Thread( "BluetoothServer" ),
|
|
mpCommunicators( pCommunicators )
|
|
{
|
|
}
|
|
|
|
BluetoothServer::~BluetoothServer()
|
|
{
|
|
}
|
|
|
|
void BluetoothServer::execute()
|
|
{
|
|
#if defined(LINUX) && defined(ENABLE_DBUS)
|
|
g_type_init();
|
|
|
|
GError *aError = NULL;
|
|
|
|
DBusGConnection *aConnection = NULL;
|
|
aConnection = dbus_g_bus_get( DBUS_BUS_SYSTEM, &aError );
|
|
|
|
if ( aError != NULL ) {
|
|
g_error_free (aError);
|
|
return;
|
|
}
|
|
|
|
|
|
DBusGProxy *aManager = NULL;
|
|
aManager = dbus_g_proxy_new_for_name( aConnection, "org.bluez", "/",
|
|
"org.bluez.Manager" );
|
|
|
|
if ( aManager == NULL )
|
|
{
|
|
dbus_g_connection_unref( aConnection );
|
|
return;
|
|
}
|
|
|
|
gboolean aResult;
|
|
// The following should be a DBusGObjectPath, however the necessary
|
|
// typedef is missing in older version of dbus-glib.
|
|
char *aAdapterPath = NULL;
|
|
aResult = dbus_g_proxy_call( aManager, "DefaultAdapter", &aError,
|
|
G_TYPE_INVALID,
|
|
DBUS_TYPE_G_OBJECT_PATH, &aAdapterPath,
|
|
G_TYPE_INVALID);
|
|
|
|
g_object_unref( G_OBJECT( aManager ));
|
|
if ( !aResult )
|
|
{
|
|
dbus_g_connection_unref( aConnection );
|
|
return;
|
|
}
|
|
|
|
DBusGProxy *aAdapter = NULL;
|
|
aAdapter = dbus_g_proxy_new_for_name( aConnection, "org.bluez",
|
|
aAdapterPath, "org.bluez.Service" );
|
|
g_free( aAdapterPath );
|
|
if ( aAdapter == NULL )
|
|
{
|
|
dbus_g_connection_unref( aConnection );
|
|
return;
|
|
}
|
|
|
|
// Add the record -- the handle can be used to release it again, but we
|
|
// don't bother as the record is automatically released when LO exits.
|
|
guint aHandle;
|
|
aResult = dbus_g_proxy_call( aAdapter, "AddRecord", &aError,
|
|
G_TYPE_STRING, BLUETOOTH_SERVICE_RECORD ,
|
|
G_TYPE_INVALID,
|
|
G_TYPE_UINT, &aHandle,
|
|
G_TYPE_INVALID);
|
|
|
|
g_object_unref( G_OBJECT( aAdapter ));
|
|
dbus_g_connection_unref( aConnection );
|
|
if ( !aResult)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// ---------------- Socket code
|
|
int aSocket;
|
|
if ( (aSocket = socket( AF_BLUETOOTH, SOCK_STREAM, BTPROTO_RFCOMM )) < 0 )
|
|
{
|
|
return;
|
|
}
|
|
|
|
sockaddr_rc aAddr;
|
|
aAddr.rc_family = AF_BLUETOOTH;
|
|
// BDADDR_ANY is broken, so use memset to set to 0.
|
|
memset( &aAddr.rc_bdaddr, 0, sizeof( aAddr.rc_bdaddr ) );
|
|
aAddr.rc_channel = 5;
|
|
|
|
if ( bind( aSocket, (sockaddr*) &aAddr, sizeof(aAddr)) < 0 ) {
|
|
close( aSocket );
|
|
return;
|
|
}
|
|
|
|
if ( listen( aSocket, 1 ) < 0 )
|
|
{
|
|
close( aSocket );
|
|
return;
|
|
}
|
|
|
|
sockaddr_rc aRemoteAddr;
|
|
socklen_t aRemoteAddrLen = sizeof(aRemoteAddr);
|
|
while ( true )
|
|
{
|
|
int bSocket;
|
|
if ( (bSocket = accept(aSocket, (sockaddr*) &aRemoteAddr, &aRemoteAddrLen)) < 0 )
|
|
{
|
|
close( aSocket );
|
|
return;
|
|
} else {
|
|
Communicator* pCommunicator = new Communicator( new BufferedStreamSocket( bSocket) );
|
|
mpCommunicators->push_back( pCommunicator );
|
|
pCommunicator->launch();
|
|
}
|
|
}
|
|
|
|
#endif // LINUX && ENABLE_DBUS
|
|
|
|
#ifdef WIN32
|
|
|
|
#endif
|
|
|
|
#ifdef MACOSX
|
|
|
|
#endif
|
|
}
|
|
|
|
|
|
BluetoothServer *sd::BluetoothServer::spServer = NULL;
|
|
|
|
void BluetoothServer::setup( std::vector<Communicator*>* pCommunicators )
|
|
{
|
|
if (spServer)
|
|
return;
|
|
|
|
spServer = new BluetoothServer( pCommunicators );
|
|
spServer->launch();
|
|
}
|
|
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|