Re: DBus LEAdvertisement and Python

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hello Luiz,

Thank you for your patience! I have something working (details below)

On Wed, 24 Jul 2019 at 11:45, Luiz Augusto von Dentz
<luiz.dentz@xxxxxxxxx> wrote:
>
> Hi Barry,
>
> On Wed, Jul 24, 2019 at 12:41 PM Barry Byford <31baz66@xxxxxxxxx> wrote:
> >
> > I have exhausted my ideas for debugging this issue. Using d-feet and
> > various other utilities the creation of the beacon using python-dbus
> > and pydbus looks identical. However, BlueZ doesn't seem to be able to
> > parse the properties from LEAdvertisement1 when it is created with
> > pydbus.
> >
> > It would be great if anyone has any ideas for what else to try.
> > Have people had similar issues using DBus libraries in other languages?
> >
> > For reference, I have installed pydbus with "sudo pip3 install pydbus"
> > I am running with BlueZ 5.50.
> >
> > The test code I am using is:
> >
> > <python3 code>
> > import pydbus
> > from gi.repository import GLib
> >
> >
> > class Advertisement:
> >     """
> >     <node>
> >         <interface name='org.bluez.LEAdvertisement1'>
> >             <method name='Release'>
> >                 <annotation name="org.freedesktop.DBus.Method.NoReply"
> > value="true"/>
> >             </method>
> >             <annotation
> > name="org.freedesktop.DBus.Properties.PropertiesChanged"
> > value="const"/>
> >             <property name="Type" type="s" access="read"/>
> >             <property name="ServiceUUIDs" type="as" access="read"/>
> >             <property name="ManufacturerData" type="a{sv}" access="read"/>
>
> Type here is wrong, it should be a{qv} see:
>
> https://git.kernel.org/pub/scm/bluetooth/bluez.git/tree/client/advertising.c#n464
>
> Im surprised the daemon don't generate report an error? Or it does
> register but it still doesn't advertise?
>
> >             <property name="SolicitUUIDs" type="as" access="read"/>
> >             <property name="ServiceData" type="a{sv}" access="read"/>
> >             <property name="IncludeTxPower" type="b" access="read"/>
> >         </interface>
> >     </node>
> >     """
> >     LE_ADVERTISEMENT_IFACE = 'org.bluez.LEAdvertisement1'
> >
> >     def Release(self):
> >         pass
> >
> >     @property
> >     def Type(self):
> >         return 'broadcast'
> >
> >     @property
> >     def SolicitUUIDs(self):
> >         return []
>
> Don't include if empty.
>
> >     @property
> >     def ServiceUUIDs(self):
> >         return ['FEAA']
> >
> >     @property
> >     def ServiceData(self):
> >         return {'FEAA': GLib.Variant('ay', [0x10, 0x08, 0x03, 0x75,
> > 0x6B, 0x42, 0x61,
> >                          0x7A, 0x2e, 0x67, 0x69, 0x74, 0x68, 0x75,
> >                          0x62, 0x2E, 0x69, 0x6F])}
> >
> >     @property
> >     def IncludeTxPower(self):
> >         return False
> >
> >     @property
> >     def ManufacturerData(self):
> >         return []
>
> Same here, don't include it empty.
>
> >
> > class LEAdvertisement:
> >     def __init__(self, service, object_path):
> >         bus = pydbus.SystemBus()
> >         bname = bus.request_name(service)
> >         reg1 = bus.register_object(object_path, Advertisement(), None)
> >
> >
> > class LEAdvertisingManager:
> >     def __init__(self, object_path):
> >         lea_iface = 'org.bluez.LEAdvertisingManager1'
> >         bus = pydbus.SystemBus()
> >         ad_manager = bus.get('org.bluez', '/org/bluez/hci0')[lea_iface]
> >         ad_manager.RegisterAdvertisement(object_path, {})
> >
> >
> > if __name__ == '__main__':
> >     app_name = 'ukBaz.bluezero'
> >     app_path = '/ukBaz/bluezero/advertisement0099'
> >     LEAdvertisement(app_name, app_path)
> >     LEAdvertisingManager(app_path)
> >
> >     loop = GLib.MainLoop()
> >     try:
> >         loop.run()
> >     except KeyboardInterrupt:
> >         print("\nStopping ...")
> >         loop.quit()
> >
> > </python code>
> >
> > I have the following config to open the permissions up for this service name:
> >
> > $ more /etc/dbus-1/system.d/ukBaz.bluezero.conf
> > <?xml version="1.0" encoding="UTF-8"?> <!-- -*- XML -*- -->
> > <!DOCTYPE busconfig PUBLIC "-//freedesktop//DTD D-BUS Bus
> > Configuration 1.0//EN"
> >  "http://www.freedesktop.org/standards/dbus/1.0/busconfig.dtd";>
> > <busconfig>
> >
> >   <!-- ../system.conf have denied everything, so we just punch some
> > holes -->
> >
> >   <policy context="default">
> >     <allow own="ukBaz.bluezero"/>
> >     <allow send_destination="ukBaz.bluezero"
> >            send_interface="org.freedesktop.DBus.Introspectable"/>
> >     <allow send_type="method_call" log="true"/>
> >
> >   </policy>
> >
> > </busconfig>
> >
> >
> > Feedback welcome.
> >
> > Cheers,
> > Barry
> >
> >
> > On Sat, 20 Jul 2019 at 17:44, Barry Byford <31baz66@xxxxxxxxx> wrote:
> > >
> > > Hello,
> > >
> > > I have done a little more investigation on this. The difference in
> > > behaviour between the working library and the failing library comes
> > > down to this line in the BlueZ source.
> > > https://git.kernel.org/pub/scm/bluetooth/bluez.git/tree/gdbus/client.c#n720
> > > which is called from:
> > > https://git.kernel.org/pub/scm/bluetooth/bluez.git/tree/src/advertising.c#n1020
> > > It appears g_hash_table_lookup is returning NULL.
> > > I am not familiar with C and have been doing this investigation with
> > > fprintf statements. I don't know how to get the values of
> > > "proxy->prop_list" and "name". Does anyone know how I can print these
> > > values out?
> > > Even better would be, if someone could point me at to what this
> > > relates to in the Python LE Advertisement setup through DBus.
> > > I've looked up the g_hash_table_lookup command:
> > > https://developer.gnome.org/glib/stable/glib-Hash-Tables.html#g-hash-table-lookup
> > > but this hasn't helped me.
> > >
> > > Thanks,
> > > Barry
> > >
> > > On Thu, 18 Jul 2019 at 22:04, Barry Byford <31baz66@xxxxxxxxx> wrote:
> > > >
> > > > Hello Luiz,
> > > >
> > > > Thanks for the prompt response. Below are some updates following your input.
> > > >
> > > > On Thu, 18 Jul 2019 at 09:26, Luiz Augusto von Dentz
> > > > <luiz.dentz@xxxxxxxxx> wrote:
> > > > >
> > > > > Hi Barry,
> > > > >
> > > > > On Thu, Jul 18, 2019 at 10:03 AM Barry Byford <31baz66@xxxxxxxxx> wrote:
> > > > > >
> > > > > > Hello,
> > > > > >
> > > > > > The Python examples provided in the test directory of the BlueZ
> > > > > > repository use the python-dbus library. Looking at the documentation
> > > > > > of that library it does start by raising concerns and offering
> > > > > > alternatives. I have been looking at some of the alternatives and am
> > > > > > having difficultly getting them to work with BlueZ. I have been
> > > > > > focusing on the org.bluez.LEAdvertisement1 interface.
> > > > > >
> > > > > > Looking at the documentation, it says the service, interface and
> > > > > > object details are:
> > > > > > Service org.bluez
> > > > > > Interface org.bluez.LEAdvertisement1
> > > > > > Object path freely definable
> > > > > >
> > > > > > I am not being successful at publishing to the org.bluez service. It
> > > > > > is also not where the current examples publish to.
> > > > > > Looking at the GattProfile1 documentation, it has service and object
> > > > > > as application dependant.
> > > > > > Service <application dependent>
> > > > > > Interface org.bluez.GattProfile1
> > > > > > Object path <application dependent>
> > > > > > Should the documentation of Service on LEAdvertisement1 be freely
> > > > > > definable also?
> > > > >
> > > > > Yep, the bus name is up to the application which usually don't
> > > > > register a friendly name if you try to register with 'org.bluez' it
> > > > > would probably conflict with the daemon itself so you wouldn't be able
> > > > > to register that name anyway.
> > > > >
> > > > > > I have created the LEAdvertisement1 interface so that it has an
> > > > > > ObjectManager and is introspectable. When I pass the object to
> > > > > > RegisterAdvertisement on the org.bluez.LEAdvertisingManager1 interface
> > > > > > it accepts it (does not give an error) but does not register the data
> > > > > > and I see no advertisement appear. Is there somewhere I can find a
> > > > > > more detailed description of what needs to be on the
> > > > > > org.bluez.LEAdvertisement1 interface that will work with
> > > > > > RegisterAdvertisement?
> > > > >
> > > > > Do you have the bluetoothd output when you register, I get the
> > > > > following when using bluetoothctl:
> > > > >
> > > > > bluetooth]# power on
> > > > > Changing power on succeeded
> > > > > [bluetooth]# advertise on
> > > > > [CHG] Controller B8:8A:60:D8:17:D7 SupportedInstances: 0x04
> > > > > [CHG] Controller B8:8A:60:D8:17:D7 ActiveInstances: 0x01
> > > > > Advertising object registered
> > > > > Tx Power: off
> > > > > Name: off
> > > > > Apperance: off
> > > > > Discoverable: off
> > > > > [bluetooth]#
> > > >
> > > > Using the new dbus library I get:
> > > > Log from bluetoothctl:
> > > > [CHG] Controller FC:F8:AE:8F:0C:A4 SupportedInstances: 0x04
> > > > [CHG] Controller FC:F8:AE:8F:0C:A4 ActiveInstances: 0x01
> > > >
> > > > From the bluetoothd log:
> > > > bluetoothd[2856]: src/advertising.c:register_advertisement()
> > > > RegisterAdvertisement
> > > > bluetoothd[2856]: src/advertising.c:client_create() Adding proxy for
> > > > /ukBaz/bluezero/advertisement1
> > > > bluetoothd[2856]: src/advertising.c:register_advertisement()
> > > > Registered advertisement at path /ukBaz/bluezero/advertisement1
> > > > bluetoothd[2856]: src/advertising.c:refresh_adv() Refreshing
> > > > advertisement: /ukBaz/bluezero/advertisement1
> > > > bluetoothd[2856]: src/advertising.c:add_adv_callback() Advertisement
> > > > registered: /ukBaz/bluezero/advertisement1
> > > >
> > > >
> > > > With the old pyton-dbus library (that is working) I get:
> > > > log from bluetoothctl:
> > > > [CHG] Controller FC:F8:AE:8F:0C:A4 SupportedInstances: 0x04
> > > > [CHG] Controller FC:F8:AE:8F:0C:A4 ActiveInstances: 0x01
> > > >
> > > > From the bluetoothd log:
> > > > bluetoothd[2856]: src/advertising.c:register_advertisement()
> > > > RegisterAdvertisement
> > > > bluetoothd[2856]: src/advertising.c:client_create() Adding proxy for
> > > > /ukBaz/bluezero/advertisement0001
> > > > bluetoothd[2856]: src/advertising.c:register_advertisement()
> > > > Registered advertisement at path /ukBaz/bluezero/advertisement0001
> > > > bluetoothd[2856]: src/advertising.c:parse_service_uuids() Adding
> > > > ServiceUUID: FEAA
> > > > bluetoothd[2856]: src/advertising.c:parse_service_data() Adding
> > > > ServiceData for FEAA
> > > > bluetoothd[2856]: src/advertising.c:refresh_adv() Refreshing
> > > > advertisement: /ukBaz/bluezero/advertisement0001
> > > > bluetoothd[2856]: src/advertising.c:add_adv_callback() Advertisement
> > > > registered: /ukBaz/bluezero/advertisement0001
> > > >
> > > >
> > > > >
> > > > > bluetoothd[6103]: src/advertising.c:client_create() Adding proxy for
> > > > > /org/bluez/advertising
> > > > > bluetoothd[6103]: src/advertising.c:register_advertisement()
> > > > > Registered advertisement at path /org/bluez/advertising
> > > > > bluetoothd[6103]: src/advertising.c:refresh_adv() Refreshing
> > > > > advertisement: /org/bluez/advertising
> > > > > bluetoothd[6103]: src/advertising.c:add_adv_callback() Advertisement
> > > > > registered: /org/bluez/advertising
> > > > >
> > > > >
> > > > > > For reference, I have put below what I am currently  putting on the
> > > > > > org.bluez.LEAdvertisement1 interface.
> > > > > >
> > > > > > Thanks,
> > > > > > Barry
> > > > > >
> > > > > >
> > > > > > $ busctl call ukBaz.bluezero /ukBaz/bluezero/advertisement1
> > > > > > org.freedesktop.DBus.ObjectManager GetManagedObjects
> > > > > > a{oa{sa{sv}}} 1 "/ukBaz/bluezero/advertisement1" 5
> > > > > > "org.freedesktop.DBus.Properties" 0
> > > > > > "org.freedesktop.DBus.Introspectable" 0 "org.freedesktop.DBus.Peer" 0
> > > > > > "org.freedesktop.DBus.ObjectManager" 0 "org.bluez.LEAdvertisement1" 6
> > > > > > "Type" s "broadcast" "ServiceUUIDs" as 0 "ManufacturerData" a{sv} 0
> > > > > > "SolicitUUIDs" as 0 "ServiceData" a{sv} 1 "FEAA" ay 18 16 8 3 117 107
> > > > > > 66 97 122 46 103 105 116 104 117 98 46 105 111 "IncludeTxPower" b
> > > > > > false
> > > > >
> > > > > Not sure if that is the problem but usually ObjectManager is suppose
> > > > > to be on the '/' (root) path.
> > > >
> > > > Good point. It is not clear (to me anyway) exactly what the
> > > > requirement is from reading the spec
> > > > https://dbus.freedesktop.org/doc/dbus-specification.html#standard-interfaces-objectmanager
> > > > I have moved the object manager to the root (/) but this does not seem
> > > > to have changed anything.
> > > >
> > > > $ busctl call ukBaz.bluezero / org.freedesktop.DBus.ObjectManager
> > > > GetManagedObjects
> > > > a{oa{sa{sv}}} 1 "/ukBaz/bluezero/advertisement1" 4
> > > > "org.freedesktop.DBus.Properties" 0
> > > > "org.freedesktop.DBus.Introspectable" 0 "org.freedesktop.DBus.Peer" 0
> > > > "org.bluez.LEAdvertisement1" 6 "Type" s "broadcast" "ServiceUUIDs" as
> > > > 0 "ManufacturerData" a{sv} 0 "SolicitUUIDs" as 0 "ServiceData" a{sv} 1
> > > > "FEAA" ay 18 16 8 3 117 107 66 97 122 46 103 105 116 104 117 98 46 105
> > > > 111 "IncludeTxPower" b false
> > > >
> > > >
> > > > >
> > > > > > $ busctl call ukBaz.bluezero /ukBaz/bluezero/advertisement1
> > > > > > org.freedesktop.DBus.Properties GetAll s org.bluez.LEAdvertisement1
> > > > > > a{sv} 6 "Type" s "broadcast" "ServiceUUIDs" as 0 "ManufacturerData"
> > > > > > a{sv} 0 "SolicitUUIDs" as 0 "ServiceData" a{sv} 1 "FEAA" ay 18 16 8 3
> > > > > > 117 107 66 97 122 46 103 105 116 104 117 98 46 105 111
> > > > > > "IncludeTxPower" b false
> > > > > >
> > > >
> > > > Using the d-feet application I did a GetAll on the advertisements and
> > > > the data looks identical for both
> > > > Using the new DBus library:
> > > > {'IncludeTxPower': False,
> > > >  'ManufacturerData': {},
> > > >  'ServiceData': {'FEAA': [16,
> > > >                           8,
> > > >                           3,
> > > >                           117,
> > > >                           107,
> > > >                           66,
> > > >                           97,
> > > >                           122,
> > > >                           46,
> > > >                           103,
> > > >                           105,
> > > >                           116,
> > > >                           104,
> > > >                           117,
> > > >                           98,
> > > >                           46,
> > > >                           105,
> > > >                           111]},
> > > >  'ServiceUUIDs': ['FEAA'],
> > > >  'SolicitUUIDs': [],
> > > >  'Type': 'broadcast'}
> > > >
> > > > Using the old Python-dbus library:
> > > > {'IncludeTxPower': False,
> > > >  'ServiceData': {'FEAA': [16,
> > > >                           8,
> > > >                           3,
> > > >                           117,
> > > >                           107,
> > > >                           66,
> > > >                           97,
> > > >                           122,
> > > >                           46,
> > > >                           103,
> > > >                           105,
> > > >                           116,
> > > >                           104,
> > > >                           117,
> > > >                           98,
> > > >                           46,
> > > >                           105,
> > > >                           111]},
> > > >  'ServiceUUIDs': ['FEAA'],
> > > >  'Type': 'broadcast'}
> > > >
> > > > > > $ busctl call ukBaz.bluezero /ukBaz/bluezero/advertisement1
> > > > > > org.freedesktop.DBus.Introspectable Introspect
> > > > > > s "<!DOCTYPE node PUBLIC \"-//freedesktop//DTD D-BUS Object
> > > > > > Introspection 1.0//EN\"\n
> > > > > > \"http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd\";>\n<!--
> > > > > > GDBus 2.60.4 -->\n<node>\n  <interface
> > > > > > name=\"org.freedesktop.DBus.Properties\">\n    <method name=\"Get\">\n
> > > > > >      <arg type=\"s\" name=\"interface_name\" direction=\"in\"/>\n
> > > > > > <arg type=\"s\" name=\"property_name\" direction=\"in\"/>\n      <arg
> > > > > > type=\"v\" name=\"value\" direction=\"out\"/>\n    </method>\n
> > > > > > <method name=\"GetAll\">\n      <arg type=\"s\"
> > > > > > name=\"interface_name\" direction=\"in\"/>\n      <arg type=\"a{sv}\"
> > > > > > name=\"properties\" direction=\"out\"/>\n    </method>\n    <method
> > > > > > name=\"Set\">\n      <arg type=\"s\" name=\"interface_name\"
> > > > > > direction=\"in\"/>\n      <arg type=\"s\" name=\"property_name\"
> > > > > > direction=\"in\"/>\n      <arg type=\"v\" name=\"value\"
> > > > > > direction=\"in\"/>\n    </method>\n    <signal
> > > > > > name=\"PropertiesChanged\">\n      <arg type=\"s\"
> > > > > > name=\"interface_name\"/>\n      <arg type=\"a{sv}\"
> > > > > > name=\"changed_properties\"/>\n      <arg type=\"as\"
> > > > > > name=\"invalidated_properties\"/>\n    </signal>\n  </interface>\n
> > > > > > <interface name=\"org.freedesktop.DBus.Introspectable\">\n    <method
> > > > > > name=\"Introspect\">\n      <arg type=\"s\" name=\"xml_data\"
> > > > > > direction=\"out\"/>\n    </method>\n  </interface>\n  <interface
> > > > > > name=\"org.freedesktop.DBus.Peer\">\n    <method name=\"Ping\"/>\n
> > > > > > <method name=\"GetMachineId\">\n      <arg type=\"s\"
> > > > > > name=\"machine_uuid\" direction=\"out\"/>\n    </method>\n
> > > > > > </interface>\n  <interface
> > > > > > name=\"org.freedesktop.DBus.ObjectManager\">\n    <method
> > > > > > name=\"GetManagedObjects\">\n      <arg type=\"a{oa{sa{sv}}}\"
> > > > > > name=\"object_paths_interfaces_and_properties\" direction=\"out\">\n
> > > > > >    </arg>\n    </method>\n    <signal name=\"InterfacesAdded\">\n
> > > > > > <arg type=\"o\" name=\"object_path\">\n      </arg>\n      <arg
> > > > > > type=\"a{sa{sv}}\" name=\"interfaces_and_properties\">\n      </arg>\n
> > > > > >    </signal>\n    <signal name=\"InterfacesRemoved\">\n      <arg
> > > > > > type=\"o\" name=\"object_path\">\n      </arg>\n      <arg type=\"as\"
> > > > > > name=\"interfaces\">\n      </arg>\n    </signal>\n  </interface>\n
> > > > > > <interface name=\"org.bluez.LEAdvertisement1\">\n    <annotation
> > > > > > name=\"org.freedesktop.DBus.Properties.PropertiesChanged\"
> > > > > > value=\"const\">\n    </annotation>\n    <method name=\"Release\">\n
> > > > > >    <annotation name=\"org.freedesktop.DBus.Method.NoReply\"
> > > > > > value=\"true\">\n      </annotation>\n    </method>\n    <property
> > > > > > type=\"s\" name=\"Type\" access=\"read\">\n    </property>\n
> > > > > > <property type=\"as\" name=\"ServiceUUIDs\" access=\"read\">\n
> > > > > > </property>\n    <property type=\"a{sv}\" name=\"ManufacturerData\"
> > > > > > access=\"read\">\n    </property>\n    <property type=\"as\"
> > > > > > name=\"SolicitUUIDs\" access=\"read\">\n    </property>\n    <property
> > > > > > type=\"a{sv}\" name=\"ServiceData\" access=\"read\">\n
> > > > > > </property>\n    <property type=\"b\" name=\"IncludeTxPower\"
> > > > > > access=\"read\">\n    </property>\n  </interface>\n</node>\n"
> > > > >
> > > > >
> > > > >
> > > > > --
> > > > > Luiz Augusto von Dentz

I have had it explained to me that there is a blocking call in the
pydbus library and there is a need to have two threads running.
With that addition to the code I am able to create a beacon using the
pydbus library.
As this topic has come up before on the mailing list it seems
appropriate to share the code I used to test this but make no claim
this is the only or best solution.

Thanks again.

<python3 code>
import threading
import pydbus
from gi.repository import GLib


class Beacon:
    """
    <node>
        <interface name='org.bluez.LEAdvertisement1'>
            <method name='Release'>
                <annotation name="org.freedesktop.DBus.Method.NoReply"
value="true"/>
            </method>
            <annotation
name="org.freedesktop.DBus.Properties.PropertiesChanged"
value="const"/>
            <property name="Type" type="s" access="read"/>
            <property name="ServiceUUIDs" type="as" access="read"/>
            <property name="ServiceData" type="a{sv}" access="read"/>
            <property name="IncludeTxPower" type="b" access="read"/>
            <property name="ManufacturerData" type="a{qv}" access="read"/>
            <property name="SolicitUUIDs" type="as" access="read"/>
        </interface>
    </node>
    """
    LE_ADVERTISEMENT_IFACE = 'org.bluez.LEAdvertisement1'

    def Release(self):
        pass

    @property
    def Type(self):
        return 'broadcast'

    @property
    def ServiceUUIDs(self):
        return ['FEAA']

    @property
    def ServiceData(self):
        return {'FEAA': pydbus.Variant('ay', [0x10, 0x08, 0x03, 0x75, 0x6B,
                                              0x42, 0x61, 0x7A, 0x2e, 0x67,
                                              0x69, 0x74, 0x68, 0x75, 0x62,
                                              0x2E, 0x69, 0x6F])}

    @property
    def IncludeTxPower(self):
        return False

    @property
    def ManufacturerData(self):
        return []

    @property
    def SolicitUUIDs(self):
        return []


class LEAdvertisement:
    def __init__(self, service, object_path):
        bus = pydbus.SystemBus()
        bname = bus.request_name(service)
        reg1 = bus.register_object(object_path, Beacon(), None)


class LEAdvertisingManager:
    def __init__(self, object_path):
        lea_iface = 'org.bluez.LEAdvertisingManager1'
        bus = pydbus.SystemBus()
        ad_manager = bus.get('org.bluez', '/org/bluez/hci0')[lea_iface]
        ad_manager.RegisterAdvertisement(object_path, {})
        print('Registered Ad')


def publish_now():
    print('Publishing Ad')
    aloop = GLib.MainLoop()
    aloop.run()


def thread_function():
    print('Starting thread')
    LEAdvertisement(app_name, app_path)
    publish_now()
    print('thread finished')


if __name__ == '__main__':
    app_name = 'ukBaz.bluezero'
    app_path = '/ukBaz/bluezero/advertisement0099'

    loop = GLib.MainLoop()
    x = threading.Thread(target=thread_function, daemon=True)
    x.start()
    LEAdvertisingManager(app_path)

    try:
        loop.run()
    except KeyboardInterrupt:
        print("\nStopping ...")
        loop.quit()


>
>
>
> --
> Luiz Augusto von Dentz



[Index of Archives]     [Bluez Devel]     [Linux Wireless Networking]     [Linux Wireless Personal Area Networking]     [Linux ATH6KL]     [Linux USB Devel]     [Linux Media Drivers]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Big List of Linux Books]

  Powered by Linux