I'm working on CoreBluetooth and have a problem with discoverCharacteristics
method.
I initialized my services and characteristics so:
m_centralManager = [[CBCentralManager alloc] initWithDelegate:self queue:nil];
CBUUID *uuid = [CBUUID UUIDWithString:ServiceUUID];
m_services = [NSArray arrayWithObject:uuid];
CBMutableCharacteristic *characteristics = [CBMutableCharacteristic alloc];
characteristics = [characteristics initWithType:uuid
properties:CBCharacteristicPropertyWriteWithoutResponse
value:nil
permissions:CBAttributePermissionsWriteable];
m_characteristics = [NSArray arrayWithObject:characteristics];
and my ServiceUUID
is @"ffe0"
- (void) centralManagerDidUpdateState:(CBCentralManager *)central
{
m_scanLabel = [UILabel new];
[self addLabel:m_scanLabel topPadding:TextFieldFirstPadding];
if (central.state == CBCentralManagerStatePoweredOn)
{
m_scanLabel.text = @"Launch scan";
m_timer = [NSTimer scheduledTimerWithTimeInterval:1.0
target:self
selector:@selector(scan)
userInfo:nil
repeats:YES];
}
else
{
m_scanLabel.text = @"Error connection. Check bluetooth state";
[m_timer invalidate];
}
[m_scanLabel sizeToFit];
}
- (void) centralManager:(CBCentralManager *)central didDiscoverPeripheral:(CBPeripheral *)peripheral advertisementData:(NSDictionary *)advertisementData RSSI:(NSNumber *)RSSI
{
// todo check right peripheral
m_peripheralLabel = [UILabel new];
[self addLabel:m_peripheralLabel topPadding:CGRectGetMaxY(m_scanLabel.frame)];
[m_timer invalidate];
if (m_peripheral == nil)
{
m_peripheral = peripheral;
m_peripheralLabel.text = @"Connecting...";
[m_peripheralLabel sizeToFit];
[m_centralManager connectPeripheral:peripheral options:nil];
}
else
{
[m_centralManager retrieveConnectedPeripheralsWithServices:m_services];
}
}
- (void) centralManager:(CBCentralManager *)central didConnectPeripheral:(CBPeripheral *)peripheral
{
[m_centralManager stopScan];
peripheral.delegate = self;
m_didConnectLabel = [UILabel new];
[self addLabel:m_didConnectLabel topPadding:CGRectGetMaxY(m_peripheralLabel.frame)];
m_didConnectLabel.text = @"Connected. Launch discover";
[m_didConnectLabel sizeToFit];
[peripheral discoverServices:m_services];
}
- (void) centralManager:(CBCentralManager *)central didDisconnectPeripheral:(CBPeripheral *)peripheral error:(NSError *)error
{
m_timer = [NSTimer scheduledTimerWithTimeInterval:1.0
target:self
selector:@selector(scan)
userInfo:nil
repeats:YES];
}
- (void) peripheral:(CBPeripheral *)peripheral didDiscoverServices:(NSError *)error
{
for (CBService *service in peripheral.services)
{
[peripheral discoverCharacteristics:m_characteristics forService:service];
}
}
The app crashes in this place: [peripheral discoverCharacteristics:m_characteristics forService:service];
With a log like this (here I used 1802 service, but it has the same results):
* thread #1: tid = 0x1710ca, 0x0000000199e15d3c libobjc.A.dylib`objc_exception_throw, queue = 'com.apple.main-thread, stop reason = breakpoint 1.1
frame #0: 0x0000000199e15d3c libobjc.A.dylib`objc_exception_throw
frame #1: 0x000000018d9befdc CoreFoundation`+[NSException raise:format:] + 128
frame #2: 0x000000018d688464 CoreBluetooth`-[CBXpcConnection allocXpcObjectWithNSObject:] + 556
frame #3: 0x000000018d688590 CoreBluetooth`-[CBXpcConnection allocXpcArrayWithNSArray:] + 252
frame #4: 0x000000018d6883e0 CoreBluetooth`-[CBXpcConnection allocXpcObjectWithNSObject:] + 424
frame #5: 0x000000018d6887b4 CoreBluetooth`-[CBXpcConnection allocXpcDictionaryWithNSDictionary:] + 360
frame #6: 0x000000018d687b48 CoreBluetooth`-[CBXpcConnection allocXpcMsg:args:] + 104
frame #7: 0x000000018d687bd4 CoreBluetooth`-[CBXpcConnection sendAsyncMsg:args:] + 32
frame #8: 0x000000018d687c20 CoreBluetooth`-[CBXpcConnection sendMsg:args:] + 36
frame #9: 0x000000010002daa0 SampleApp`-[ViewController peripheral:didDiscoverServices:](self=0x0000000126e04900, _cmd=0x00000001913f147c, peripheral=0x00000001700bb120, error=0x0000000000000000) + 364 at ViewController.m:121
frame #10: 0x000000018d682c08 CoreBluetooth`-[CBPeripheral handleServicesDiscovered:] + 756
frame #11: 0x000000018d67ed88 CoreBluetooth`-[CBCentralManager xpcConnection:didReceiveMsg:args:] + 136
frame #12: 0x000000018d687e38 CoreBluetooth`__34-[CBXpcConnection handleMsg:args:]_block_invoke + 84
frame #13: 0x000000019a3e4420 libdispatch.dylib`_dispatch_call_block_and_release + 24
frame #14: 0x000000019a3e43e0 libdispatch.dylib`_dispatch_client_callout + 16
frame #15: 0x000000019a3e756c libdispatch.dylib`_dispatch_main_queue_callback_4CF + 344
frame #16: 0x000000018d97ed64 CoreFoundation`__CFRUNLOOP_IS_SERVICING_THE_MAIN_DISPATCH_QUEUE__ + 12
frame #17: 0x000000018d97d0a4 CoreFoundation`__CFRunLoopRun + 1452
frame #18: 0x000000018d8bdb38 CoreFoundation`CFRunLoopRunSpecific + 452
frame #19: 0x000000019329f830 GraphicsServices`GSEventRunModal + 168
frame #20: 0x00000001908fc0e8 UIKit`UIApplicationMain + 1156
frame #21: 0x000000010002e7e0 SampleApp`main(argc=1, argv=0x000000016fdd7ca8) + 116 at main.m:16
frame #22: 0x000000019a3ffaa0 libdyld.dylib`start + 4
What I'm doing wrong?
The problem is your m_characteristics that you are passing to peripheral discoverCharacteristics
is an array of CBMutableCharacteristics, when it should be an array of CBUUIDs. So, you should have
m_characteristics = [NSArray arrayWithObject:uuid];
where uuid is a CBUUID created from your characteristic uuid string.