diff options
author | Michael Meeks <michael.meeks@suse.com> | 2013-02-18 13:31:21 +0000 |
---|---|---|
committer | Michael Meeks <michael.meeks@suse.com> | 2013-02-19 09:12:15 +0000 |
commit | e8f2870be12b8886cb58f5c35f607c00ebe96ea3 (patch) | |
tree | 8ae851006bc331e4242efc3e50510f5c5b78f625 | |
parent | 033a2c71fd9a35481f13fcd9c1e2d0339437baf0 (diff) |
sdremote: re-work discoverable property get/set to use libdbus.
Change-Id: I5e11a0fcfd8db648a40c738cbc8181889cb36136
-rw-r--r-- | sd/source/ui/remotecontrol/BluetoothServer.cxx | 438 | ||||
-rw-r--r-- | sd/source/ui/remotecontrol/BluetoothServer.hxx | 8 |
2 files changed, 248 insertions, 198 deletions
diff --git a/sd/source/ui/remotecontrol/BluetoothServer.cxx b/sd/source/ui/remotecontrol/BluetoothServer.cxx index 29027a519c52..164b81322cff 100644 --- a/sd/source/ui/remotecontrol/BluetoothServer.cxx +++ b/sd/source/ui/remotecontrol/BluetoothServer.cxx @@ -21,20 +21,12 @@ #ifdef LINUX_BLUETOOTH #include <glib.h> #include <dbus/dbus.h> - #include <dbus/dbus-glib.h> #include <errno.h> #include <fcntl.h> #include <sys/unistd.h> #include <sys/socket.h> #include <bluetooth/bluetooth.h> #include <bluetooth/rfcomm.h> - #define DBUS_TYPE_G_STRING_ANY_HASHTABLE (dbus_g_type_get_map( "GHashTable", G_TYPE_STRING, G_TYPE_VALUE )) - #ifndef G_VALUE_INIT - #define G_VALUE_INIT {0,{{0}}} // G_VALUE_INIT only present in glib >= 2.30 - #endif - #ifndef DBusGObjectPath - #define DBusGObjectPath char // DBusGObjectPath is only present in newer version of dbus-glib - #endif #include "BluetoothServiceRecord.hxx" #include "BufferedStreamSocket.hxx" #endif @@ -75,26 +67,52 @@ using namespace sd; #ifdef LINUX_BLUETOOTH -struct sd::BluetoothServerImpl { - // the glib mainloop running in the thread - GMainContext *mpContext; - volatile bool mbExitMainloop; -}; - struct DBusObject { OString maBusName; OString maPath; OString maInterface; + DBusObject() { } DBusObject( const char *pBusName, const char *pPath, const char *pInterface ) - : maBusName( pBusName ), maPath( pPath ), maInterface( pInterface ) - { } + : maBusName( pBusName ), maPath( pPath ), maInterface( pInterface ) { } DBusMessage *getMethodCall( const char *pName ) { return dbus_message_new_method_call( maBusName.getStr(), maPath.getStr(), maInterface.getStr(), pName ); } + DBusObject *cloneForInterface( const char *pInterface ) + { + DBusObject *pObject = new DBusObject(); + + pObject->maBusName = maBusName; + pObject->maPath = maPath; + pObject->maInterface = pInterface; + + return pObject; + } +}; + +struct sd::BluetoothServerImpl { + // the glib mainloop running in the thread + GMainContext *mpContext; + DBusConnection *mpConnection; + DBusObject *mpService; + volatile bool mbExitMainloop; + + BluetoothServerImpl() : + mpContext( g_main_context_new() ), + mpConnection( NULL ), + mpService( NULL ), + mbExitMainloop( false ) + { } + + DBusObject *getAdapter() + { + if( !mpService ) + return NULL; + return mpService->cloneForInterface( "org.bluez.Adapter" ); + } }; static DBusConnection * @@ -116,19 +134,15 @@ dbusConnectToNameOnBus() return pConnection; } -static DBusObject * -bluezGetDefaultAdapter( DBusConnection *pConnection, - const gchar* pInterfaceType = "org.bluez.Adapter" ) +static DBusMessage * +sendUnrefAndWaitForReply( DBusConnection *pConnection, DBusMessage *pMsg ) { - DBusMessage *pMsg; - DBusMessageIter it; - DBusPendingCall *pPending; + DBusPendingCall *pPending = NULL; - pMsg = DBusObject( "org.bluez", "/", "org.bluez.Manager" ).getMethodCall( "DefaultAdapter" ); if( !pMsg || !dbus_connection_send_with_reply( pConnection, pMsg, &pPending, -1 /* default timeout */ ) ) { - SAL_WARN( "sdremote.bluetooth", "Memory allocation failed on message" ); + SAL_WARN( "sdremote.bluetooth", "Memory allocation failed on message send" ); dbus_message_unref( pMsg ); return NULL; } @@ -138,14 +152,25 @@ bluezGetDefaultAdapter( DBusConnection *pConnection, dbus_pending_call_block( pPending ); // block for reply pMsg = dbus_pending_call_steal_reply( pPending ); - if( !pMsg || !dbus_message_iter_init( pMsg, &it ) ) - { + if( !pMsg ) SAL_WARN( "sdremote.bluetooth", "no valid reply / timeout" ); - dbus_pending_call_unref( pPending ); - return NULL; - } + dbus_pending_call_unref( pPending ); + return pMsg; +} + +static DBusObject * +bluezGetDefaultService( DBusConnection *pConnection ) +{ + DBusMessage *pMsg; + DBusMessageIter it; + const gchar* pInterfaceType = "org.bluez.Service"; + pMsg = DBusObject( "org.bluez", "/", "org.bluez.Manager" ).getMethodCall( "DefaultAdapter" ); + pMsg = sendUnrefAndWaitForReply( pConnection, pMsg ); + + if(!pMsg || !dbus_message_iter_init( pMsg, &it ) ) + return NULL; if( DBUS_TYPE_OBJECT_PATH != dbus_message_iter_get_arg_type( &it ) ) SAL_WARN( "sdremote.bluetooth", "invalid type of reply to DefaultAdapter: '" @@ -169,25 +194,13 @@ bluezRegisterServiceRecord( DBusConnection *pConnection, DBusObject *pAdapter, { DBusMessage *pMsg; DBusMessageIter it; - DBusPendingCall *pPending; pMsg = pAdapter->getMethodCall( "AddRecord" ); dbus_message_iter_init_append( pMsg, &it ); dbus_message_iter_append_basic( &it, DBUS_TYPE_STRING, &pServiceRecord ); - if(!dbus_connection_send_with_reply( pConnection, pMsg, &pPending, - -1 /* default timeout */ ) ) - if( !dbus_connection_send( pConnection, pMsg, NULL ) ) - { - SAL_WARN( "sdremote.bluetooth", "failed to send service record" ); - return false; - } - dbus_connection_flush( pConnection ); - dbus_message_unref( pMsg ); - - dbus_pending_call_block( pPending ); // block for reply + pMsg = sendUnrefAndWaitForReply( pConnection, pMsg ); - pMsg = dbus_pending_call_steal_reply( pPending ); if( !pMsg || !dbus_message_iter_init( pMsg, &it ) || dbus_message_iter_get_arg_type( &it ) != DBUS_TYPE_UINT32 ) { @@ -198,8 +211,6 @@ bluezRegisterServiceRecord( DBusConnection *pConnection, DBusObject *pAdapter, // We ignore the uint de-registration handle we get back: // bluez will clean us up automatically on exit - dbus_pending_call_unref( pPending ); - return true; } @@ -244,7 +255,12 @@ bluezCreateListeningSocket() return nSocket; } -#endif // defined(LINUX) && defined(ENABLE_DBUS) +#endif // LINUX_BLUETOOTH + +void BluetoothServer::addCommunicator( Communicator* pCommunicator ) +{ + mpCommunicators->push_back( pCommunicator ); +} #if defined(MACOSX) @@ -394,195 +410,222 @@ void incomingCallback( void *userRefCon, #endif // MACOSX - -BluetoothServer::BluetoothServer( std::vector<Communicator*>* pCommunicators ) - : meWasDiscoverable( UNKNOWN ), - mpImpl( NULL ), - mpCommunicators( pCommunicators ) -{ #ifdef LINUX_BLUETOOTH - g_type_init(); - mpImpl = new BluetoothServerImpl(); - mpImpl->mpContext = g_main_context_new(); - mpImpl->mbExitMainloop = false; -#endif -} - -BluetoothServer::~BluetoothServer() -{ -} - - -void BluetoothServer::ensureDiscoverable() -{ - if( !spServer || spServer->meWasDiscoverable != UNKNOWN ) - return; - bool bDiscoverable = spServer->isDiscoverable(); - spServer->meWasDiscoverable = bDiscoverable ? DISCOVERABLE : NOT_DISCOVERABLE; - spServer->setDiscoverable( true ); -} - -void BluetoothServer::restoreDiscoverable() -{ - if(!spServer) - return; - - if ( spServer->meWasDiscoverable == NOT_DISCOVERABLE ) - spServer->setDiscoverable( false ); - spServer->meWasDiscoverable = UNKNOWN; +extern "C" { + static gboolean ensureDiscoverable_cb(gpointer) + { + BluetoothServer::doEnsureDiscoverable(); + return FALSE; // remove source + } + static gboolean restoreDiscoverable_cb(gpointer) + { + BluetoothServer::doRestoreDiscoverable(); + return FALSE; // remove source + } } -bool BluetoothServer::isDiscoverable() +static bool +getBooleanProperty( DBusConnection *pConnection, DBusObject *pAdapter, + const char *pPropertyName, bool *pBoolean ) { -#if 0 // (defined(LINUX) && !defined(__FreeBSD_kernel__)) && defined(ENABLE_DBUS) - SAL_INFO( "sdremote.bluetooth", "BluetoothServer::isDiscoverable called" ); - g_type_init(); - gboolean aResult; + *pBoolean = false; - GError *aError = NULL; + if( !pAdapter ) + return false; - DBusGConnection *aConnection = NULL; - aConnection = dbus_g_bus_get( DBUS_BUS_SYSTEM, &aError ); + DBusMessage *pMsg; + pMsg = sendUnrefAndWaitForReply( pConnection, + pAdapter->getMethodCall( "GetProperties" ) ); - if ( aError != NULL ) { - g_error_free (aError); - SAL_INFO( "sdremote.bluetooth", "did not get DBusGConnection" ); + DBusMessageIter it; + if( !pMsg || !dbus_message_iter_init( pMsg, &it ) ) + { + SAL_WARN( "sdremote.bluetooth", "no valid reply / timeout" ); return false; } - DBusGProxy* aAdapter = bluezGetDefaultAdapter( pConnection ); - if ( aAdapter == NULL ) + if( DBUS_TYPE_ARRAY != dbus_message_iter_get_arg_type( &it ) ) { - dbus_g_connection_unref( aConnection ); - SAL_INFO( "sdremote.bluetooth", "did not get default adaptor" ); + SAL_WARN( "sdremote.bluetooth", "no valid reply / timeout" ); return false; } - GHashTable* aProperties = NULL; - aResult = dbus_g_proxy_call( aAdapter, "GetProperties", &aError, - G_TYPE_INVALID, - DBUS_TYPE_G_STRING_ANY_HASHTABLE, &aProperties, - G_TYPE_INVALID); - g_object_unref( G_OBJECT( aAdapter )); - dbus_g_connection_unref( aConnection ); - if ( !aResult || aError ) + DBusMessageIter arrayIt; + dbus_message_iter_recurse( &it, &arrayIt ); + + while( dbus_message_iter_get_arg_type( &arrayIt ) == DBUS_TYPE_DICT_ENTRY ) { - if ( aError ) - g_error_free( aError ); - SAL_INFO( "sdremote.bluetooth", "did not get properties" ); - return false; - } + DBusMessageIter dictIt; + dbus_message_iter_recurse( &arrayIt, &dictIt ); - gboolean aIsDiscoverable = g_value_get_boolean( (GValue*) g_hash_table_lookup( - aProperties, "Discoverable" ) ); + const char *pName = NULL; + if( dbus_message_iter_get_arg_type( &dictIt ) == DBUS_TYPE_STRING ) + { + dbus_message_iter_get_basic( &dictIt, &pName ); + if( pName != NULL && !strcmp( pName, pPropertyName ) ) + { + SAL_INFO( "sdremote.bluetooth", "hit " << pPropertyName << " property" ); + dbus_message_iter_next( &dictIt ); + dbus_bool_t bBool = false; - g_hash_table_unref( aProperties ); + if( dbus_message_iter_get_arg_type( &dictIt ) == DBUS_TYPE_VARIANT ) + { + DBusMessageIter variantIt; + dbus_message_iter_recurse( &dictIt, &variantIt ); + + if( dbus_message_iter_get_arg_type( &variantIt ) == DBUS_TYPE_BOOLEAN ) + { + dbus_message_iter_get_basic( &variantIt, &bBool ); + SAL_INFO( "sdremote.bluetooth", "" << pPropertyName << " is " << bBool ); + *pBoolean = bBool; + return true; + } + else + SAL_WARN( "sdremote.bluetooth", "" << pPropertyName << " type " << + dbus_message_iter_get_arg_type( &variantIt ) ); + } + else + SAL_WARN( "sdremote.bluetooth", "variant type ? " << + dbus_message_iter_get_arg_type( &dictIt ) ); + } + else + { + const char *pStr = pName ? pName : "<null>"; + SAL_INFO( "sdremote.bluetooth", "property '" << pStr << "'" ); + } + } + else + SAL_WARN( "sdremote.bluetooth", "unexpected property key type " + << dbus_message_iter_get_arg_type( &dictIt ) ); + dbus_message_iter_next( &arrayIt ); + } + dbus_message_unref( pMsg ); - SAL_INFO( "sdremote.bluetooth", "BluetoothServer::isDiscoverable() returns " << static_cast< bool >( aIsDiscoverable ) ); - return aIsDiscoverable; -#else // defined(LINUX) && defined(ENABLE_DBUS) return false; -#endif } -void BluetoothServer::setDiscoverable( bool bDiscoverable ) +static void +setDiscoverable( DBusConnection *pConnection, DBusObject *pAdapter, bool bDiscoverable ) { -#if 0 // (defined(LINUX) && !defined(__FreeBSD_kernel__)) && defined(ENABLE_DBUS) - SAL_INFO( "sdremote.bluetooth", "BluetoothServer::setDiscoverable called" ); - g_type_init(); - gboolean aResult; + SAL_INFO( "sdremote.bluetooth", "setDiscoverable to " << bDiscoverable ); - GError *aError = NULL; + bool bPowered = false; + if( !getBooleanProperty( pConnection, pAdapter, "Powered", &bPowered ) || !bPowered ) + return; // nothing to do - DBusGConnection *aConnection = NULL; - aConnection = dbus_g_bus_get( DBUS_BUS_SYSTEM, &aError ); + DBusMessage *pMsg; + DBusMessageIter it, varIt; - if ( aError != NULL ) - { - g_error_free (aError); - return; - } + // set timeout to zero + pMsg = pAdapter->getMethodCall( "SetProperty" ); + dbus_message_iter_init_append( pMsg, &it ); + const char *pTimeoutStr = "DiscoverableTimeout"; + dbus_message_iter_append_basic( &it, DBUS_TYPE_STRING, &pTimeoutStr ); + dbus_message_iter_open_container( &it, DBUS_TYPE_VARIANT, + DBUS_TYPE_UINT32_AS_STRING, &varIt ); + dbus_uint32_t nTimeout = 0; + dbus_message_iter_append_basic( &varIt, DBUS_TYPE_UINT32, &nTimeout ); + dbus_message_iter_close_container( &it, &varIt ); + dbus_connection_send( pConnection, pMsg, NULL ); // async send - why not ? + dbus_message_unref( pMsg ); - DBusGProxy* aAdapter = bluezGetDefaultAdapter( pConnection ); - if ( aAdapter == NULL ) - { - dbus_g_connection_unref( aConnection ); - return; - } + // set discoverable value + pMsg = pAdapter->getMethodCall( "SetProperty" ); + dbus_message_iter_init_append( pMsg, &it ); + const char *pDiscoverableStr = "Discoverable"; + dbus_message_iter_append_basic( &it, DBUS_TYPE_STRING, &pDiscoverableStr ); + dbus_message_iter_open_container( &it, DBUS_TYPE_VARIANT, + DBUS_TYPE_BOOLEAN_AS_STRING, &varIt ); + dbus_bool_t bValue = bDiscoverable; + dbus_message_iter_append_basic( &varIt, DBUS_TYPE_BOOLEAN, &bValue ); + dbus_message_iter_close_container( &it, &varIt ); // async send - why not ? + dbus_connection_send( pConnection, pMsg, NULL ); + dbus_message_unref( pMsg ); +} - GHashTable* aProperties; - aResult = dbus_g_proxy_call( aAdapter, "GetProperties", &aError, - G_TYPE_INVALID, - DBUS_TYPE_G_STRING_ANY_HASHTABLE, &aProperties, - G_TYPE_INVALID); +#endif // LINUX_BLUETOOTH - if ( !aResult || aError ) - { - SAL_WARN( "sdremote.bluetooth", "GetProperties failed" ); - if ( aError ) - { - g_error_free( aError ); - SAL_WARN( "sdremote.bluetooth", "with error " << aError->message ); - } +BluetoothServer::BluetoothServer( std::vector<Communicator*>* pCommunicators ) + : meWasDiscoverable( UNKNOWN ), + mpImpl( NULL ), + mpCommunicators( pCommunicators ) +{ +#ifdef LINUX_BLUETOOTH + mpImpl = new BluetoothServerImpl(); +#endif +} + +BluetoothServer::~BluetoothServer() +{ +} + +void BluetoothServer::ensureDiscoverable() +{ +#ifdef LINUX_BLUETOOTH + // Push it all across into our mainloop + if( !spServer ) return; - } + GSource *pIdle = g_idle_source_new(); + g_source_set_callback( pIdle, ensureDiscoverable_cb, NULL, NULL ); + g_source_set_priority( pIdle, G_PRIORITY_DEFAULT ); + g_source_attach( pIdle, spServer->mpImpl->mpContext ); + g_source_unref( pIdle ); +#endif +} - gboolean aPowered = g_value_get_boolean( (GValue*) g_hash_table_lookup( - aProperties, "Powered" ) ); +void BluetoothServer::restoreDiscoverable() +{ +#ifdef LINUX_BLUETOOTH + // Push it all across into our mainloop + if( !spServer ) + return; + GSource *pIdle = g_idle_source_new(); + g_source_set_callback( pIdle, restoreDiscoverable_cb, NULL, NULL ); + g_source_set_priority( pIdle, G_PRIORITY_DEFAULT_IDLE ); + g_source_attach( pIdle, spServer->mpImpl->mpContext ); + g_source_unref( pIdle ); +#endif +} - g_hash_table_unref( aProperties ); - if ( !aPowered ) - { - SAL_INFO( "sdremote.bluetooth", "Bluetooth adapter not powered, returning" ); - g_object_unref( G_OBJECT( aAdapter )); +void BluetoothServer::doEnsureDiscoverable() +{ + if (!spServer->mpImpl->mpConnection || + spServer->meWasDiscoverable != UNKNOWN ) return; - } - GValue aTimeout = G_VALUE_INIT; - g_value_init( &aTimeout, G_TYPE_UINT ); - g_value_set_uint( &aTimeout, 0 ); - aResult = dbus_g_proxy_call( aAdapter, "SetProperty", &aError, - G_TYPE_STRING, "DiscoverableTimeout", - G_TYPE_VALUE, &aTimeout, G_TYPE_INVALID, G_TYPE_INVALID); - if ( !aResult || aError ) - { - SAL_WARN( "sdremote.bluetooth", "SetProperty(DiscoverableTimeout) failed" ); - if ( aError ) - { - g_error_free( aError ); - SAL_WARN( "sdremote.bluetooth", "with error " << aError->message ); - } +#ifdef LINUX_BLUETOOTH + // Find out if we are discoverable already ... + DBusObject *pAdapter = spServer->mpImpl->getAdapter(); + if( !pAdapter ) return; - } - GValue bDiscoverableGValue = G_VALUE_INIT; - g_value_init( &bDiscoverableGValue, G_TYPE_BOOLEAN ); - g_value_set_boolean( &bDiscoverableGValue, bDiscoverable ); - aResult = dbus_g_proxy_call( aAdapter, "SetProperty", &aError, - G_TYPE_STRING, "Discoverable", - G_TYPE_VALUE, &bDiscoverableGValue, G_TYPE_INVALID, G_TYPE_INVALID); - if ( !aResult || aError ) + bool bDiscoverable; + if( getBooleanProperty( spServer->mpImpl->mpConnection, pAdapter, + "Discoverable", &bDiscoverable ) ) { - SAL_WARN( "sdremote.bluetooth", "SetProperty(Discoverable) failed" ); - if ( aError ) - { - g_error_free( aError ); - SAL_WARN( "sdremote.bluetooth", "with error " << aError->message ); - } - return; + spServer->meWasDiscoverable = bDiscoverable ? DISCOVERABLE : NOT_DISCOVERABLE; + if( !bDiscoverable ) + setDiscoverable( spServer->mpImpl->mpConnection, pAdapter, true ); } - g_object_unref( G_OBJECT( aAdapter )); - dbus_g_connection_unref( aConnection ); -#else // defined(LINUX) && defined(ENABLE_DBUS) - (void) bDiscoverable; // avoid warnings + delete pAdapter; #endif } -void BluetoothServer::addCommunicator( Communicator* pCommunicator ) +void BluetoothServer::doRestoreDiscoverable() { - mpCommunicators->push_back( pCommunicator ); + if( spServer->meWasDiscoverable == NOT_DISCOVERABLE ) + { +#ifdef LINUX_BLUETOOTH + DBusObject *pAdapter = spServer->mpImpl->getAdapter(); + if( !pAdapter ) + return; + setDiscoverable( spServer->mpImpl->mpConnection, pAdapter, false ); + delete pAdapter; +#endif + } + spServer->meWasDiscoverable = UNKNOWN; } void SAL_CALL BluetoothServer::run() @@ -594,14 +637,15 @@ void SAL_CALL BluetoothServer::run() if( !pConnection ) return; - DBusObject *pService = bluezGetDefaultAdapter( pConnection, "org.bluez.Service" ); - if( !pService ) + mpImpl->mpService = bluezGetDefaultService( pConnection ); + if( !mpImpl->mpService ) { dbus_connection_unref( pConnection ); return; } - if( !bluezRegisterServiceRecord( pConnection, pService, bluetooth_service_record ) ) + if( !bluezRegisterServiceRecord( pConnection, mpImpl->mpService, + bluetooth_service_record ) ) return; int nSocket = bluezCreateListeningSocket(); @@ -633,6 +677,8 @@ void SAL_CALL BluetoothServer::run() else SAL_WARN( "sdremote.bluetooth", "failed to poll for incoming dbus signals" ); + mpImpl->mpConnection = pConnection; + while( !mpImpl->mbExitMainloop ) { aDBusFD.revents = 0; @@ -661,6 +707,10 @@ void SAL_CALL BluetoothServer::run() } } + g_main_context_unref( mpImpl->mpContext ); + mpImpl->mpConnection = NULL; + mpImpl->mpContext = NULL; + #elif defined(WIN32) WORD wVersionRequested; WSADATA wsaData; diff --git a/sd/source/ui/remotecontrol/BluetoothServer.hxx b/sd/source/ui/remotecontrol/BluetoothServer.hxx index b922d6094901..24b8dec36ba2 100644 --- a/sd/source/ui/remotecontrol/BluetoothServer.hxx +++ b/sd/source/ui/remotecontrol/BluetoothServer.hxx @@ -28,15 +28,15 @@ namespace sd /// restore the state of discoverability from before ensureDiscoverable static void restoreDiscoverable(); - void addCommunicator( Communicator* pCommunicator ); + // called by C / idle callbacks + static void doEnsureDiscoverable(); + static void doRestoreDiscoverable(); + void addCommunicator( Communicator* pCommunicator ); private: BluetoothServer( std::vector<Communicator*>* pCommunicators ); ~BluetoothServer(); - bool isDiscoverable(); - void setDiscoverable( bool bDiscoverable ); - enum { UNKNOWN, DISCOVERABLE, NOT_DISCOVERABLE } meWasDiscoverable; static BluetoothServer *spServer; |