Skip to content

Commit b278978

Browse files
authored
[Linux] Reuse AdapterIterator in all places where listing managed objects (#32372)
* Use dedicated iterator for listing BlueZ objects * Use GAutoPtr to manage GDBusObjectManager instance * Wrap static function with lambda * Fix iterator initialization for AdapterIterator
1 parent 610435a commit b278978

8 files changed

+58
-89
lines changed

src/platform/GLibTypeDeleter.h

+6
Original file line numberDiff line numberDiff line change
@@ -120,6 +120,12 @@ struct GAutoPtrDeleter<GDBusConnection>
120120
using deleter = GObjectDeleter;
121121
};
122122

123+
template <>
124+
struct GAutoPtrDeleter<GDBusObjectManager>
125+
{
126+
using deleter = GObjectDeleter;
127+
};
128+
123129
template <>
124130
struct GAutoPtrDeleter<GDBusObjectManagerServer>
125131
{

src/platform/Linux/bluez/AdapterIterator.cpp

+17-47
Original file line numberDiff line numberDiff line change
@@ -26,69 +26,38 @@ namespace chip {
2626
namespace DeviceLayer {
2727
namespace Internal {
2828

29-
AdapterIterator::~AdapterIterator()
30-
{
31-
if (mManager != nullptr)
32-
{
33-
g_object_unref(mManager);
34-
}
35-
36-
if (mObjectList != nullptr)
37-
{
38-
g_list_free_full(mObjectList, g_object_unref);
39-
}
40-
}
41-
42-
CHIP_ERROR AdapterIterator::Initialize(AdapterIterator * self)
29+
CHIP_ERROR AdapterIterator::Initialize()
4330
{
4431
// When creating D-Bus proxy object, the thread default context must be initialized. Otherwise,
4532
// all D-Bus signals will be delivered to the GLib global default main context.
4633
VerifyOrDie(g_main_context_get_thread_default() != nullptr);
4734

48-
CHIP_ERROR err = CHIP_NO_ERROR;
4935
GAutoPtr<GError> error;
50-
51-
self->mManager = g_dbus_object_manager_client_new_for_bus_sync(
36+
mManager.reset(g_dbus_object_manager_client_new_for_bus_sync(
5237
G_BUS_TYPE_SYSTEM, G_DBUS_OBJECT_MANAGER_CLIENT_FLAGS_NONE, BLUEZ_INTERFACE, "/",
5338
bluez_object_manager_client_get_proxy_type, nullptr /* unused user data in the Proxy Type Func */,
54-
nullptr /* destroy notify */, nullptr /* cancellable */, &error.GetReceiver());
39+
nullptr /* destroy notify */, nullptr /* cancellable */, &error.GetReceiver()));
5540

56-
VerifyOrExit(self->mManager != nullptr, ChipLogError(DeviceLayer, "Failed to get DBUS object manager for listing adapters.");
57-
err = CHIP_ERROR_INTERNAL);
41+
VerifyOrReturnError(mManager, CHIP_ERROR_INTERNAL,
42+
ChipLogError(DeviceLayer, "Failed to get D-Bus object manager for listing adapters: %s", error->message));
5843

59-
self->mObjectList = g_dbus_object_manager_get_objects(self->mManager);
60-
self->mCurrentListItem = self->mObjectList;
44+
mObjectList.Init(mManager.get());
45+
mIterator = mObjectList.begin();
6146

62-
exit:
63-
if (error != nullptr)
64-
{
65-
ChipLogError(DeviceLayer, "DBus error: %s", error->message);
66-
}
67-
68-
return err;
47+
return CHIP_NO_ERROR;
6948
}
7049

7150
bool AdapterIterator::Advance()
7251
{
73-
if (mCurrentListItem == nullptr)
52+
for (; mIterator != BluezObjectList::end(); ++mIterator)
7453
{
75-
return false;
76-
}
77-
78-
while (mCurrentListItem != nullptr)
79-
{
80-
BluezAdapter1 * adapter = bluez_object_get_adapter1(BLUEZ_OBJECT(mCurrentListItem->data));
81-
if (adapter == nullptr)
54+
BluezAdapter1 * adapter = bluez_object_get_adapter1(&(*mIterator));
55+
if (adapter != nullptr)
8256
{
83-
mCurrentListItem = mCurrentListItem->next;
84-
continue;
57+
mCurrentAdapter.reset(adapter);
58+
++mIterator;
59+
return true;
8560
}
86-
87-
mCurrentAdapter.reset(adapter);
88-
89-
mCurrentListItem = mCurrentListItem->next;
90-
91-
return true;
9261
}
9362

9463
return false;
@@ -112,9 +81,10 @@ uint32_t AdapterIterator::GetIndex() const
11281

11382
bool AdapterIterator::Next()
11483
{
115-
if (mManager == nullptr)
84+
if (!mManager)
11685
{
117-
CHIP_ERROR err = PlatformMgrImpl().GLibMatterContextInvokeSync(Initialize, this);
86+
CHIP_ERROR err = PlatformMgrImpl().GLibMatterContextInvokeSync(
87+
+[](AdapterIterator * self) { return self->Initialize(); }, this);
11888
VerifyOrReturnError(err == CHIP_NO_ERROR, false, ChipLogError(DeviceLayer, "Failed to initialize adapter iterator"));
11989
}
12090

src/platform/Linux/bluez/AdapterIterator.h

+7-7
Original file line numberDiff line numberDiff line change
@@ -17,13 +17,15 @@
1717

1818
#pragma once
1919

20-
#include <string>
20+
#include <cstdint>
2121

2222
#include <gio/gio.h>
2323

2424
#include <lib/core/CHIPError.h>
25+
#include <platform/GLibTypeDeleter.h>
2526
#include <platform/Linux/dbus/bluez/DbusBluez.h>
2627

28+
#include "BluezObjectList.h"
2729
#include "Types.h"
2830

2931
namespace chip {
@@ -46,8 +48,6 @@ namespace Internal {
4648
class AdapterIterator
4749
{
4850
public:
49-
~AdapterIterator();
50-
5151
/// Moves to the next DBUS interface.
5252
///
5353
/// MUST be called before any of the 'current value' methods are
@@ -65,17 +65,17 @@ class AdapterIterator
6565

6666
private:
6767
/// Sets up the DBUS manager and loads the list
68-
static CHIP_ERROR Initialize(AdapterIterator * self);
68+
CHIP_ERROR Initialize();
6969

7070
/// Loads the next value in the list.
7171
///
7272
/// Returns true if a value could be loaded, false if no more items to
7373
/// iterate through.
7474
bool Advance();
7575

76-
GDBusObjectManager * mManager = nullptr; // DBus connection
77-
GList * mObjectList = nullptr; // listing of objects on the bus
78-
GList * mCurrentListItem = nullptr; // current item viewed in the list
76+
GAutoPtr<GDBusObjectManager> mManager;
77+
BluezObjectList mObjectList;
78+
BluezObjectIterator mIterator;
7979
// Data valid only if Next() returns true
8080
GAutoPtr<BluezAdapter1> mCurrentAdapter;
8181
};

src/platform/Linux/bluez/BluezConnection.cpp

+5-12
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@
3434
#include <system/SystemPacketBuffer.h>
3535

3636
#include "BluezEndpoint.h"
37+
#include "BluezObjectList.h"
3738
#include "Types.h"
3839

3940
namespace chip {
@@ -95,10 +96,6 @@ BluezConnection::ConnectionDataBundle::ConnectionDataBundle(const BluezConnectio
9596

9697
CHIP_ERROR BluezConnection::Init(const BluezEndpoint & aEndpoint)
9798
{
98-
// populate the service and the characteristics
99-
GList * objects = nullptr;
100-
GList * l;
101-
10299
if (!aEndpoint.mIsCentral)
103100
{
104101
mpService = reinterpret_cast<BluezGattService1 *>(g_object_ref(aEndpoint.mpService));
@@ -107,11 +104,9 @@ CHIP_ERROR BluezConnection::Init(const BluezEndpoint & aEndpoint)
107104
}
108105
else
109106
{
110-
objects = g_dbus_object_manager_get_objects(aEndpoint.mpObjMgr);
111-
112-
for (l = objects; l != nullptr; l = l->next)
107+
for (BluezObject & object : BluezObjectList(aEndpoint.mpObjMgr))
113108
{
114-
BluezGattService1 * service = bluez_object_get_gatt_service1(BLUEZ_OBJECT(l->data));
109+
BluezGattService1 * service = bluez_object_get_gatt_service1(&object);
115110
if (service != nullptr)
116111
{
117112
if ((BluezIsServiceOnDevice(service, mpDevice)) == TRUE &&
@@ -126,9 +121,9 @@ CHIP_ERROR BluezConnection::Init(const BluezEndpoint & aEndpoint)
126121

127122
VerifyOrExit(mpService != nullptr, ChipLogError(DeviceLayer, "FAIL: NULL service in %s", __func__));
128123

129-
for (l = objects; l != nullptr; l = l->next)
124+
for (BluezObject & object : BluezObjectList(aEndpoint.mpObjMgr))
130125
{
131-
BluezGattCharacteristic1 * char1 = bluez_object_get_gatt_characteristic1(BLUEZ_OBJECT(l->data));
126+
BluezGattCharacteristic1 * char1 = bluez_object_get_gatt_characteristic1(&object);
132127
if (char1 != nullptr)
133128
{
134129
if ((BluezIsCharOnService(char1, mpService) == TRUE) &&
@@ -164,8 +159,6 @@ CHIP_ERROR BluezConnection::Init(const BluezEndpoint & aEndpoint)
164159
}
165160

166161
exit:
167-
if (objects != nullptr)
168-
g_list_free_full(objects, g_object_unref);
169162
return CHIP_NO_ERROR;
170163
}
171164

src/platform/Linux/bluez/BluezEndpoint.cpp

+6-7
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,7 @@
7272
#include <system/SystemPacketBuffer.h>
7373

7474
#include "BluezConnection.h"
75+
#include "BluezObjectList.h"
7576
#include "Types.h"
7677

7778
namespace chip {
@@ -420,15 +421,14 @@ BluezGattService1 * BluezEndpoint::CreateGattService(const char * aUUID)
420421
return service;
421422
}
422423

423-
void BluezEndpoint::SetupAdapter()
424+
CHIP_ERROR BluezEndpoint::SetupAdapter()
424425
{
425426
char expectedPath[32];
426427
snprintf(expectedPath, sizeof(expectedPath), BLUEZ_PATH "/hci%u", mAdapterId);
427428

428-
GList * objects = g_dbus_object_manager_get_objects(mpObjMgr);
429-
for (auto l = objects; l != nullptr && mAdapter.get() == nullptr; l = l->next)
429+
for (BluezObject & object : BluezObjectList(mpObjMgr))
430430
{
431-
GAutoPtr<BluezAdapter1> adapter(bluez_object_get_adapter1(BLUEZ_OBJECT(l->data)));
431+
GAutoPtr<BluezAdapter1> adapter(bluez_object_get_adapter1(&object));
432432
if (adapter.get() != nullptr)
433433
{
434434
if (mpAdapterAddr == nullptr) // no adapter address provided, bind to the hci indicated by nodeid
@@ -450,7 +450,7 @@ void BluezEndpoint::SetupAdapter()
450450
}
451451
}
452452

453-
VerifyOrExit(mAdapter.get() != nullptr, ChipLogError(DeviceLayer, "FAIL: NULL mAdapter in %s", __func__));
453+
VerifyOrReturnError(mAdapter, CHIP_ERROR_INTERNAL, ChipLogError(DeviceLayer, "FAIL: NULL mAdapter in %s", __func__));
454454

455455
bluez_adapter1_set_powered(mAdapter.get(), TRUE);
456456

@@ -459,8 +459,7 @@ void BluezEndpoint::SetupAdapter()
459459
// and the flag is necessary to force using LE transport.
460460
bluez_adapter1_set_discoverable(mAdapter.get(), FALSE);
461461

462-
exit:
463-
g_list_free_full(objects, g_object_unref);
462+
return CHIP_NO_ERROR;
464463
}
465464

466465
BluezConnection * BluezEndpoint::GetBluezConnection(const char * aPath)

src/platform/Linux/bluez/BluezEndpoint.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -85,7 +85,7 @@ class BluezEndpoint
8585
private:
8686
CHIP_ERROR StartupEndpointBindings();
8787

88-
void SetupAdapter();
88+
CHIP_ERROR SetupAdapter();
8989
void SetupGattServer(GDBusConnection * aConn);
9090
void SetupGattService();
9191

src/platform/Linux/bluez/BluezObjectIterator.h

+2
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@
1717

1818
#pragma once
1919

20+
#include <iterator>
21+
2022
#include <glib.h>
2123

2224
#include <platform/CHIPDeviceConfig.h>

src/platform/Linux/bluez/BluezObjectList.h

+14-15
Original file line numberDiff line numberDiff line change
@@ -35,27 +35,26 @@ namespace Internal {
3535
class BluezObjectList
3636
{
3737
public:
38-
explicit BluezObjectList(GDBusObjectManager * manager) { Initialize(manager); }
38+
BluezObjectList() = default;
39+
explicit BluezObjectList(GDBusObjectManager * manager) { Init(manager); }
3940

40-
~BluezObjectList() { g_list_free_full(mObjectList, g_object_unref); }
41-
42-
BluezObjectIterator begin() const { return BluezObjectIterator(mObjectList); }
43-
BluezObjectIterator end() const { return BluezObjectIterator(); }
44-
45-
protected:
46-
BluezObjectList() {}
47-
48-
void Initialize(GDBusObjectManager * manager)
41+
~BluezObjectList()
4942
{
50-
if (manager == nullptr)
51-
{
52-
ChipLogError(DeviceLayer, "Manager is NULL in %s", __func__);
53-
return;
54-
}
43+
if (mObjectList != nullptr)
44+
g_list_free_full(mObjectList, g_object_unref);
45+
}
5546

47+
CHIP_ERROR Init(GDBusObjectManager * manager)
48+
{
49+
VerifyOrReturnError(manager != nullptr, CHIP_ERROR_INVALID_ARGUMENT,
50+
ChipLogError(DeviceLayer, "Manager is NULL in %s", __func__));
5651
mObjectList = g_dbus_object_manager_get_objects(manager);
52+
return CHIP_NO_ERROR;
5753
}
5854

55+
BluezObjectIterator begin() const { return BluezObjectIterator(mObjectList); }
56+
static BluezObjectIterator end() { return BluezObjectIterator(); }
57+
5958
private:
6059
GList * mObjectList = nullptr;
6160
};

0 commit comments

Comments
 (0)