Peripheral transmit queue issue

2020-07-18 10:47发布

I’m working on a simple wrapper around CoreBluetooth to send any data to any device. During developing I encountered a lot of bugs in framework, they were very annoying and to make my wrapper stable I had to shorten some of functionality for reliability.

For now I’m working on sending data from peripheral.

Ok, so I have following case:

  • Client asks for value of dynamic characteristic
  • I get a callback on server-side - peripheral:didReceiveReadRequest:.

Note : I need to respond to this CBATTRequest in this method - I can’t store it elsewhere and respond to it asynchronously. (Im just putting some chunk @“PrepareToReceiveValue” that will be ignored on central side. All sending is done in queue.)

  • For providing data for various devices I constructed a queue with BTMessage's in it. (So for readRequest I create message and add it to sending queue. If chunk sending failed - I will get a callback from peripheral manager about readyToUpdateSubscribers and will ask queue to resend failed chunk)
  • So when I’m requesting immediately a lot of dynamic characteristic values and sending data from peripheral to central concurrently sometimes it just freezes sending progress and leads to disconnection.

After several testing I found out that it was all about transmit queue: If transmit queue is full and you will receive read request - it just won’t respond to it.

So I have potential unstable system state:

  1. Peripheral is sending data to some central.
  2. In my sending method updateValue:forCharac… returns NO because transmit queue is full.
  3. At this moment central requests dynamic value for characteristic and peripheral:didReceiveReadRequest: invocation will be added to current runloop.
  4. After returning from sending method it will dequeue peripheral:didReceiveReadRequest: method and responding to this request will have no effect (transmit queue is full).
  5. So in this case respondToRequest: is ignored like I didn’t invoked it at all.
  6. CoreBluetooth will not be able to send/receive any data until I will respond to request. That was the reason for freezing any sending/receiving progress with concomitant disconnection.
  7. As I mentioned before - I must respond to request in appropriate method - otherwise it will also have no effect. (Im saying it because I’ve tried to put those request in array if queue is full and respond to them when it will have some space but with no luck).

Im waiting for your proposals/suggestions how to resolve this problem, any help would be appreciated.

0条回答
登录 后发表回答