繁体   English   中英

Windows上的蓝牙低功耗低速率?

[英]Bluetooth low energy low rate on Windows?

我有一台带有自定义服务的设备,该设备使用BLE通知功能以很高的速率发送传感器数据。

我在Windows 10计算机上使用以下API: https : //msdn.microsoft.com/zh-cn/library/windows/hardware/jj159880(v=vs.85).aspx

我正在使用SetupDi API通过自定义服务ID搜索设备,然后使用CreateFile将其“连接”到该设备。

当我第一次将设备与Windows配对时,它会立即在“蓝牙设置”窗口中显示“已连接”,然后在运行我的应用程序时,它运行良好(我以高速率接收数据)。 如果我关闭我的应用程序,它将在“设置”窗口中将状态更改为“已配对”,而不是已连接(我认为还可以)。 当我再次打开应用程序时,它会连接并再次将“设置”中的状态更改为“已连接”,但是由于某种原因,现在我接收到的数据速率要低得多。 (数据本身是正确的)。 如果我通过单击“删除设备”在“蓝牙设置”窗口中断开连接,然后像在第一次以高速率再次使用之前一样将其重新配对。

我知道这不是设备本身的问题,因为它可以在Android和其他BLE支持的平台上正常工作。

任何想法可能导致此问题吗?

这是我正在使用的代码:

GUID serviceGuid = StringToGUID(GEM_SERVICE_GUID);

HDEVINFO info = SetupDiGetClassDevs(&guid, 0, 0, DIGCF_DEVICEINTERFACE);
SP_DEVICE_INTERFACE_DATA data;
data.cbSize = sizeof(SP_DEVICE_INTERFACE_DATA);

int i = 0;

while (SetupDiEnumDeviceInterfaces(info, NULL, &guid, i, &data))
{
    i++;
}

if (GetLastError() != ERROR_NO_MORE_ITEMS)
{
    // TODO throw
}

DWORD requiredSize;

if (!SetupDiGetDeviceInterfaceDetail(info, &data, NULL, 0, &requiredSize, NULL))
{
    if (GetLastError() != ERROR_INSUFFICIENT_BUFFER)
    {
        // TODO throw
    }
}

PSP_DEVICE_INTERFACE_DETAIL_DATA details = (PSP_DEVICE_INTERFACE_DETAIL_DATA)std::malloc(requiredSize);
details->cbSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA);

if (!SetupDiGetDeviceInterfaceDetail(info, &data, details, requiredSize, NULL, NULL))
{
    // TODO throw
}

m_service = CreateFile(details->DevicePath, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);

if (m_service == INVALID_HANDLE_VALUE)
{
    // TODO throw
    return;
}

BTH_LE_GATT_CHARACTERISTIC combinedDataChar = FindCharacteristicByUUID(m_service, COMBINED_DATA_CHAR_HANDLE);
BTH_LE_GATT_DESCRIPTOR desc = FindDescriptorByType(m_service, &combinedDataChar, ClientCharacteristicConfiguration);

BTH_LE_GATT_DESCRIPTOR_VALUE val;
RtlZeroMemory(&val, sizeof(val));
val.DescriptorType = ClientCharacteristicConfiguration;
val.ClientCharacteristicConfiguration.IsSubscribeToNotification = TRUE;

HRESULT res = BluetoothGATTSetDescriptorValue(m_service, &desc, &val, BLUETOOTH_GATT_FLAG_NONE);

if (res != S_OK)
{
    // TODO throw
}

BLUETOOTH_GATT_VALUE_CHANGED_EVENT_REGISTRATION chars;
chars.NumCharacteristics = 1;
chars.Characteristics[0] = combinedDataChar;

res = BluetoothGATTRegisterEvent(m_service, CharacteristicValueChangedEvent, &chars, OnValueChanged, NULL, &m_registrationHandle, BLUETOOTH_GATT_FLAG_NONE);

if (res != S_OK)
{
    // TODO throw
}

编辑:

OnValueChanged回调的代码:

void OnValueChanged(BTH_LE_GATT_EVENT_TYPE eventType, PVOID eventOutParameter, PVOID context)
{
    BLUETOOTH_GATT_VALUE_CHANGED_EVENT* e = (BLUETOOTH_GATT_VALUE_CHANGED_EVENT*)eventOutParameter;

    std::cout << e->CharacteristicValue->DataSize << std::endl;
}

我相信您是通过OnValueChanged回调接收数据的? 您的代码中未提供它,而问题可能出在其中。 如果您不愿提供其代码,建议您在“不良”会议期间进行以下实验:

  1. 从中删除所有代码,除了增加一些计数器以了解数据速率。
  2. 测量CPU负载。 如果接近核心负载,则说明您受CPU限制。
  3. 使用您选择的探查器来检查代码在哪里花费时间。

现在, OnValueChanged的代码可用:

  1. 高速率输出到控制台可能是瓶颈。 我建议您仅对事件进行计数,而输出则每秒钟计数一次,如下所示:

     static DWORD lastTicks = GetTickCount(); static DWORD count = 0; count++; DWORD ticksElapsed = GetTickCount() - lastTicks; if (ticksElapsed > 5000) { std::cout << "Rate: " << (double(count) / ticksElapsed) << " / sec" << std::endl; lastTicks = GetTickCount(); count = 0; } 
  2. 进行更多测试(例如,进行5次配对,每对配对之后进行5次连接)并提供事件发生率。 速率下降实际上可能与其他原因有关,而不是重新配对。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM