iOS Xcode BLE receiving different data lengths - ios

I am developing an iOS app using BLE, I have the BLE comms working but I do not seem to receive any data. The plan is to use multiple BLE UUID characteristics to send data from our ECU to the iOS mobile app. I know the BLE ECU is working as I have checked it with a BLE scanner.
I have followed this example, but still cannot get the BLE read to work correctly. Oddly if read the bytes back one by one I get the required data, but I cannot seem to find a solution to read the complete string in one go? I could use a loop to read all the data back, but surely there must be a way to read back the BLE receive buffer in one go?
https://www.freecodecamp.org/news/ultimate-how-to-bluetooth-swift-with-hardware-in-20-minutes/
Any help would be highly appreciated.
func peripheral(_ peripheral: CBPeripheral, didUpdateValueFor characteristic: CBCharacteristic, error: Error?) {
var characteristicASCIIValue = NSString()
guard characteristic == rxCharacteristic,
let characteristicValue = characteristic.value,
let ASCIIstring = NSString(data: characteristicValue, encoding: String.Encoding.utf8.rawValue) else { return }
print("Value Recieved: \((characteristicASCIIValue as String))")
if(characteristic.uuid.isEqual(CBUUIDs.ECU_UUID_A)
{
print ("Found ECU_UUID_A")
Extract_ECU_UUID_A(ASCIIstring)
}
else if(characteristic.uuid.isEqual(CBUUIDs.ECU_UUID_B)
{
print ("Found ECU_UUID_B")
Extract_ECU_UUID_B(ASCIIstring)
}
}

Related

How to stream audio and send data over Bluetooth in Swift?

I'm developing and app for iOS which needs to connect to a Bluetooth Device. I've been able to find and connect to devices near me using CoreBluetooth and GATT profile. The writing and reading of characteristics worked as well. These values will eventually be written when a slider is moved. However, now I also need to be able to stream audio to the Bluetooth Device.
When I looked around for streaming audio over bluetooth I noticed this could be done with AVFoundation by allowing output over bluetooth. However, then the device has to use A2DP profile. But this causes the app to be unable to read/write characteristics?
I got this piece of code to read, notify and write to specific characteristics.
func peripheral(_ peripheral: CBPeripheral, didDiscoverCharacteristicsFor service: CBService,
error: Error?) {
guard let characteristics = service.characteristics else { return }
for characteristic in characteristics {
print(characteristic)
if characteristic.properties.contains(.read) {
print("\(characteristic.uuid): properties contains .read")
if characteristic.uuid == batteryLevelCharacteristicCBUUID { peripheral.readValue(for: characteristic) }
}
if characteristic.properties.contains(.notify) {
print("\(characteristic.uuid): properties contains .notify")
if characteristic.uuid == batteryLevelCharacteristicCBUUID {
peripheral.setNotifyValue(true, for: characteristic)
}
}
if characteristic.properties.contains(.write) {
print("\(characteristic.uuid): properties contains .write")
if characteristic.uuid == writeTestCharacteristicCBUUID {
print("Writing characteristic found! Trying to write now.")
let string = "Hello World!"
let data = string.data(using: .utf8)!
peripheral.writeValue(data, for: characteristic, type: CBCharacteristicWriteType.withResponse)
}
}
}
}
This question seems similar but it is mentioned that it's not possible. Is possible send hexadecimal data to A2DP Bluetooth Device in swift?
However, I was thinking. If you look at a headset for example when you press buttons there has to be some data transfer. Can this somehow be used for what I want to do? But I can't seem to find any information about these signals or how to use those in my code.
So what I want is to be able to stream the audio and also send small values like percentages.
If you know any information that might help me please let me know. Thanks in advance!

sending data to bluetooth module to be read by arduino rom iOS swift 3

If i want to send data to a bluetooth module, which is connected to the Arduino, what code lines are specifically the ones I need to take notice of.
I want to send something like, the number '75' to the bluetooth module and the Arduino will read it
thanks
Bluetooth LE is very long and cumbersome with many delegates in between. The minimum path to write your data is:
Make sure you have bluetooth permission: CBCentralManagerDelegate.centralManagerDidUpdateStateand if so start scanning with scanForPeripherals
CBCentralManagerDelegate.didDiscover If this is the peripheral you want then set yourself as its delegate
CBPeripheralDelegate.peripheral:didDiscoverServices: If this is the service you want then stop scanning and discoverCharacteristics:for: service
CBPeripheralDelegate.peripheral:didDiscoverCharacteristicsFor: service If a characteristic in the array of characteristics is the one you want then:
func peripheral(_ peripheral: CBPeripheral, didDiscoverCharacteristicsFor service: CBService, error: Error?) {
guard let characteristics = service.characteristics else {
return
}
for characteristic in characteristics {
if characteristic.uuid == CBUUID(string: characteristicIdentifier) {
let value: UInt8 = 75
let data = Data(bytes: [value])
peripheral.writeValue(data, for: characteristic, type: .withResponse)
}
}
}

Read and handle input data when received

I am working on arduino project which involves it sending/receiving data from the iPhone to to an arduino through a BLE board. I am struggling with receiving data from the arduino back to the iPhone.
I have this function:
func readPosition() -> NSString? {
if self.positionCharacteristic == nil {
return nil
}
self.peripheral?.readValue(for: self.positionCharacteristic!)
if ((self.positionCharacteristic?.value) != nil) {
return NSString(data: self.positionCharacteristic!.value!, encoding:
String.Encoding.utf8.rawValue) }
return nil
}
My problem with it is that I don't understand how to use it in a way that it will immediately read and use what was sent from the arduino. How would I code my project to accomplish this? I need to receive constant data from a sensor that is attached to the arduino
The receiving UUID if needed:
let PositionCharUUID = CBUUID(string: "A9CD2F86-8661-4EB1-B132-367A3434BC90")
It is likely (but you need to check) that your characteristic supports Notify operations. In this case you can use
self.peripheral.setNotifyValue(true, for: self.positionCharacteristic)
You will then get a call to your peripheral delegate's didUpdateValue:for: method

Cannot Subscribe to Notify Characteristic in iOS CoreBluetooth

I have successfully connected to BLE devices using the same code, but this device is not letting me subscribe. It's frustrating because I cannot figure out how to get a meaningful error message.
There is a Read/Notify/Indicate characteristic that I am attempting to listen to. The code I send is:
peripheral.setNotifyValue(true, for: characteristic)
my listener is:
func peripheral(_ peripheral: CBPeripheral, didUpdateNotificationStateFor characteristic: CBCharacteristic, error: Error?) {
if ((error) != nil) {
NSLog("Error changing notification state: " + error!.localizedDescription)
}
if characteristic.isNotifying {
NSLog("Notification Updated")
}
}
The error I get is:
Error changing notification state: Unknown ATT error.
I have downloaded another app that interfaces with this device successfully, so I know that the hardware is functioning properly.
I don't know if there is some sort of encryption or other setting that needs to be sent to allow communication with this device. Any advice would be appreciated.

How to get the characteristic(s) of HM-10 Bluetooth LE in iOS

I am trying to send some data from an iOS device to the HM-10 Bluetooth LE Module connected to an arduino. Problem is after connecting to the module discoverServices doesn't return a characteristic for the service.
func peripheral(peripheral: CBPeripheral, didDiscoverServices error: NSError?) {
if(error != nil) {
print(error?.description)
}
for service in peripheral.services! {
let thisService = service as CBService
print("Service: \(thisService.description)")
print("Characteristic: \(thisService.characteristics)")
}
}
This outputs:
Service: <CBService: 0x137e84ea0, isPrimary = YES, UUID = FFE0>
Characteristic: nil
I am a beginner with this arduino stuff as well as iOS. So any suggestions would be welcome. Maybe there is a way to write to the bluetooth module without knowing the characteristic... I have no idea.
I finally managed to get the answer. Because of my rudimentary understanding of the CoreBluetooth Framework I forgot to call discoverCharacteristics in didDiscoverServices. Well, I am really learning by doing. (I somehow thought discoverServices would call discoverCharacteristics itself.)

Resources