Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

MTU Size Negotiation / Retrieval in the public API #246

Open
Nyrox opened this issue Feb 3, 2022 · 14 comments
Open

MTU Size Negotiation / Retrieval in the public API #246

Nyrox opened this issue Feb 3, 2022 · 14 comments

Comments

@Nyrox
Copy link

Nyrox commented Feb 3, 2022

Feature Description

It would be very useful to be able to negotiate or at least get the negotiated transfer size for a peripheral.
We are very aware that getting this right in a cross-platform manner is quite a pain and other libraries like Noble for NodeJS have just straight up chosen to not support this, but even a platform-specific API to retrieve it on some platforms would be better than nothing.

We have a custom serial protocol built on top of a characteristic pair and settling for the minimum MTU of 20 makes some operations unreasonably slow.

Would love some input on this, I am also open to writing some of the code (specifically for corebluetooth) if this is deemed a worthwhile feature. Thank you.

@qdot qdot removed their assignment Jan 16, 2023
@robbym
Copy link

robbym commented May 4, 2023

I gathered some information regarding this:

  • Windows - GattSession.MaxPduSize
    • Have not used.
  • macOS / iOS - CBPeripheral.maximumWriteValueLength
    • I've used peripheral.maximumWriteValueLength(for: .withoutResponse) reliably in the past. .withResponse has some issues. See here.
  • Linux
    • Does Bluez support this?
  • Android - BluetoothGatt.requestMtu
    • The way I understand it and have used it in the past, you call requestMtu with the largest you can support (e.g. 512), and it will negotiate the min(host.maxMtu, periph.maxMtu) and call onMtuChanged with the new mtu size.

Seems like Windows and macOS/iOS both automatically negotiate, whilst Android requires you to request it. We could have a bit of code that performs the request on the Android version automatically so that the btleplug interface could be consistent. No clue what to do for Linux.

@DCNick3
Copy link

DCNick3 commented May 11, 2023

Bluez seems to support MTU retrieval since version 5.62: bluez/bluez@1abf140

Relevant discussion: bluez/bluez#199

Bluez makes it a property of a characteristic, because with Bluetooth 5.2 Enhanced Attribute Protocol it can actually be different.. Probably a good idea to reflect this in btleplug API too?

@robbym
Copy link

robbym commented May 12, 2023

I find it kind of silly that they implemented it for a feature that has no wide support yet, and they neglect implementing it on the device level that every existing API uses and all peripheral devices < Bluetooth 5.2 uses.

In any case, I think there should be 2 different methods to get MTU. The device level negotiated MTU is done through the ATT_EXCHANGE_MTU_REQ PDU when using the fixed L2CAP CID for LE Attribute protocol (an unenhanced ATT bearer), which hasn't changed with the introduction of EATT. However, when creating multiple "enhanced ATT bearers" via the L2CAP_CREDIT_BASED_CONNECTION_REQ PDU (dynamic CIDs) it allows you to specify the MTU size for these channels.

@qdot
Copy link
Contributor

qdot commented Jun 24, 2023

Ok, so progress is happening, which is awesome, but we've got a few issues here.

@NuLL3rr0r
Copy link
Contributor

Any update on this? We also need to be able to retrieve/negotiate the MTU size on Linux, Android, and Windows if that's possible.

@cmorganBE
Copy link

Also pinging on this feature. It would bump performance up a bunch having the larger mtu.

@robbym
Copy link

robbym commented Feb 27, 2024

Sorry I haven't touched this in a year. Work had paused our Bluetooth project this passed year so I haven't worked on anything Bluetooth related. We should be resuming progress soon and I can continue where I left off.

@robbym
Copy link

robbym commented Mar 4, 2024

As @qdot points out in #317, there is conflict in the way something like Windows and Android gets the MTU. It would probably better to have the interface be an async method, and just immediately return in the Windows case. @MnlPhlp suggestion also has merit, we could have code automatically request the mtu on Android during the connection process to emulate what Windows (and presumably iOS) does. I need to really dive into bluez's approach and really see how they view the world.

In any case, we'd need to nail down the API before I move forward on the implementation(s). Will try to find time to see if there are other cross-platform solutions with MTU support.

@qdot
Copy link
Contributor

qdot commented Mar 4, 2024

I can reach out to the simpleble people and see, been talking to them lately anyways.

@robbym
Copy link

robbym commented Mar 10, 2024

I checked their bluez backend. Looks like they're just grabbing it from the first characteristic in the first service they see lol. Link to code

uint16_t PeripheralBase::mtu() {
    if (!is_connected()) return 0;

    for (auto bluez_service : device_->services()) {
        for (auto bluez_characteristic : bluez_service->characteristics()) {
            // The value provided by Bluez includes an extra 3 bytes from the GATT header
            // which needs to be removed.
            return bluez_characteristic->mtu() - 3;
        }
    }
    return 0;
}

@kdewald
Copy link

kdewald commented Mar 11, 2024

Yup, that's basically how it's offered by Bluez. In my opinion the parameter should have been added to the peripheral instead of the characteristic, but it is what it is.

The MTU is something that is negotiated by the peripheral where the central does its best to accommodate it, but once it's set it shouldn't be changed.

@Yongle-Fu
Copy link

Also pinging on this feature.

1 similar comment
@qianchenzhumeng
Copy link

Also pinging on this feature.

@robbym robbym mentioned this issue Nov 12, 2024
4 tasks
@robbym
Copy link

robbym commented Nov 12, 2024

I recreated my old PR here: #407
I'll be working on Linux next.
Does anyone have a template project/guide to get this working with Android? I don't have much experience with Rust on Android.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

9 participants