blob: 89dbbd91f065944658a1da74e9c1f7c04175875c [file] [log] [blame]
/*
*
* Copyright (c) 2022 Project CHIP Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "blecent.h"
#include <lib/support/CodeUtils.h>
#include <lib/support/logging/CHIPLogging.h>
#include <platform/ESP32/ChipDeviceScanner.h>
#define CHIPoBLE_SERVICE_UUID 0xFFF6
namespace chip {
namespace DeviceLayer {
namespace Internal {
namespace {
/// Retrieve CHIP device identification info from the device advertising data
bool NimbleGetChipDeviceInfo(const ble_hs_adv_fields & fields, chip::Ble::ChipBLEDeviceIdentificationInfo & deviceInfo)
{
// Check for CHIP Service UUID
if (fields.svc_data_uuid16 != NULL)
{
if (fields.svc_data_uuid16_len > 8 && fields.svc_data_uuid16[0] == 0xf6 && fields.svc_data_uuid16[1] == 0xff)
{
deviceInfo.OpCode = fields.svc_data_uuid16[2];
deviceInfo.DeviceDiscriminatorAndAdvVersion[0] = fields.svc_data_uuid16[3];
deviceInfo.DeviceDiscriminatorAndAdvVersion[1] = fields.svc_data_uuid16[4];
// vendor and product Id from adv
deviceInfo.DeviceVendorId[0] = fields.svc_data_uuid16[5];
deviceInfo.DeviceVendorId[1] = fields.svc_data_uuid16[6];
deviceInfo.DeviceProductId[0] = fields.svc_data_uuid16[7];
deviceInfo.DeviceProductId[1] = fields.svc_data_uuid16[8];
deviceInfo.AdditionalDataFlag = fields.svc_data_uuid16[9];
return true;
}
}
return false;
}
} // namespace
void ChipDeviceScanner::ReportDevice(const struct ble_hs_adv_fields & fields, const ble_addr_t & addr)
{
chip::Ble::ChipBLEDeviceIdentificationInfo deviceInfo;
if (NimbleGetChipDeviceInfo(fields, deviceInfo) == false)
{
ChipLogDetail(Ble, "Device %s does not look like a CHIP device", addr_str(addr.val));
return;
}
mDelegate->OnDeviceScanned(fields, addr, deviceInfo);
}
void ChipDeviceScanner::RemoveDevice()
{
// TODO
}
int ChipDeviceScanner::OnBleCentralEvent(struct ble_gap_event * event, void * arg)
{
ChipDeviceScanner * scanner = (ChipDeviceScanner *) arg;
switch (event->type)
{
case BLE_GAP_EVENT_DISC_COMPLETE: {
scanner->mIsScanning = false;
return 0;
}
case BLE_GAP_EVENT_DISC: {
/* Try to connect to the advertiser if it looks interesting. */
struct ble_hs_adv_fields fields;
ble_hs_adv_parse_fields(&fields, event->disc.data, event->disc.length_data);
scanner->ReportDevice(fields, event->disc.addr);
return 0;
}
}
return 0;
}
CHIP_ERROR ChipDeviceScanner::StartScan(uint16_t timeout)
{
VerifyOrReturnError(!mIsScanning, CHIP_ERROR_INCORRECT_STATE);
uint8_t ownAddrType;
struct ble_gap_disc_params discParams;
int rc;
/* Figure out address to use while advertising. */
rc = ble_hs_id_infer_auto(0, &ownAddrType);
if (rc != 0)
{
ChipLogError(DeviceLayer, "ble_hs_id_infer_auto failed: %d", rc);
return CHIP_ERROR_INTERNAL;
}
/* Set up discovery parameters. */
memset(&discParams, 0, sizeof(discParams));
/* Tell the controller to filter the duplicates. */
discParams.filter_duplicates = 1;
/* Perform passive scanning. */
discParams.passive = 1;
/* Use defaults for the rest of the parameters. */
discParams.itvl = 0;
discParams.window = 0;
discParams.filter_policy = 0;
discParams.limited = 0;
/* Start the discovery process. */
rc = ble_gap_disc(ownAddrType, (timeout * 1000), &discParams, OnBleCentralEvent, this);
if (rc != 0)
{
ChipLogError(DeviceLayer, "ble_gap_disc failed: %d", rc);
return CHIP_ERROR_INTERNAL;
}
mIsScanning = true;
return CHIP_NO_ERROR;
}
CHIP_ERROR ChipDeviceScanner::StopScan()
{
VerifyOrReturnError(mIsScanning, CHIP_NO_ERROR);
int rc = ble_gap_disc_cancel();
if (rc != 0)
{
ChipLogError(DeviceLayer, "ble_gap_disc_cancel failed: %d", rc);
return CHIP_ERROR_INTERNAL;
}
mIsScanning = false;
mDelegate->OnScanComplete();
return CHIP_NO_ERROR;
}
} // namespace Internal
} // namespace DeviceLayer
} // namespace chip