Avatar for DrewS

DrewS

Member since Jun 2021 • Last active Mar 2024
  • 2 conversations
  • 10 comments

Most recent activity

  • in Puck.js, Pixl.js and MDBT42
    Avatar for DrewS

    Ahh - ok so I had two bugs.

    The first was leaving off the interval:xx, and the second was that my delta time was based on the time between any bluetooth updates, not just the puck updates. That was dumb on my part.

    So now my puck js code:

    1. var presses = 0;
    2. NRF.setConnectionInterval(7.5)
    3. NRF.setAdvertising({},{name: "Puck", discoverable: true, manufacturer: 0x0590, manufacturerData:[presses], interval:20});
    4. function updateAdvertising() {
    5. presses++;
    6. NRF.setAdvertising({},{manufacturer: 0x0590, manufacturerData:[presses], interval:20});
    7. }
    8. updateAdvertising();
    9. setInterval(updateAdvertising, 1);

    and fixing my dumb python bug, gives me this output:

    1. Puck
    2. Manufacturer Data: 12, Delta Time: 0.01 seconds
    3. Puck
    4. Manufacturer Data: 15, Delta Time: 0.01 seconds
    5. Puck
    6. Manufacturer Data: 78, Delta Time: 0.87 seconds
    7. Puck
    8. Manufacturer Data: 82, Delta Time: 0.02 seconds
    9. Puck
    10. Manufacturer Data: 3c, Delta Time: 0.90 seconds
    11. Puck
    12. Manufacturer Data: 63, Delta Time: 0.07 seconds
    13. Puck
    14. Manufacturer Data: 68, Delta Time: 0.01 seconds
    15. Puck
    16. Manufacturer Data: 23, Delta Time: 0.89 seconds
    17. Puck
    18. Manufacturer Data: 24, Delta Time: 0.00 seconds
    19. Puck
    20. Manufacturer Data: d8, Delta Time: 0.90 seconds
    21. Puck
    22. Manufacturer Data: db, Delta Time: 0.01 seconds
    23. Puck
    24. Manufacturer Data: dc, Delta Time: 0.00 seconds
    25. Puck
    26. Manufacturer Data: 88, Delta Time: 0.88 seconds
    27. Puck
    28. Manufacturer Data: 89, Delta Time: 0.00 seconds
    29. Puck
    30. Manufacturer Data: 48, Delta Time: 0.91 seconds
    31. Puck

    And those numbers actually match what I see in terminal. I think at this point I'll pursue a direct connection instead of advertising (but if you have any other suggestions for that I'll happily give them a shot). Thx again for all the quick responses.

    • 11 comments
    • 816 views
    @DrewS replied
  • in Puck.js, Pixl.js and MDBT42
    Avatar for DrewS

    Ok - i made some progress here.

    I switched things over to use Bluetooth.println(...) and NRF.setConnectionInterval(7.5). I also consolidated all the logs into a single call.

    But then I started looking into ble advertising because for my use case, one direction communication is fine and if I could get 20 ms between accelerometer updates that would be awesome. I found some sample code in the puck.js documentation, and modified it like so:

    1. var presses = 0;
    2. NRF.setConnectionInterval(7.5)
    3. NRF.setAdvertising({},{name: "Puck", discoverable: true, manufacturer: 0x0590, manufacturerData:[presses]});
    4. function updateAdvertising() {
    5. //print (E.getTemperature())
    6. //print (E.getBattery())
    7. presses++;
    8. NRF.setAdvertising({},{manufacturer: 0x0590, manufacturerData:[presses]});
    9. }
    10. // Update advertising now
    11. updateAdvertising();
    12. // Update advertising every 20 miliseconds...
    13. setInterval(updateAdvertising, 1);
    14. setInterval(()=>presses++, 1);

    Yes, those are intentionally very fast intervals. I was just banging my head against it trying to get it to update quickly.

    on macOS, here is my python script:

    1. import asyncio
    2. import time
    3. from bleak import BleakScanner
    4. last_update_time = None
    5. def handle_advertisement(device, advertisement_data):
    6. global last_update_time
    7. current_time = time.time()
    8. if last_update_time is not None:
    9. time_since_last_update = current_time - last_update_time
    10. delta_time_str = f", Delta Time: {time_since_last_update:.2f} seconds"
    11. else:
    12. delta_time_str = ""
    13. last_update_time = current_time
    14. if advertisement_data.local_name == "Puck":
    15. print(advertisement_data.local_name)
    16. manufacturer_data = advertisement_data.manufacturer_data
    17. if 1424 in manufacturer_data:
    18. data = manufacturer_data[1424]
    19. data_str = ''.join(format(x, '02x') for x in data)
    20. print(f"Manufacturer Data: {data_str}{delta_time_str}")
    21. async def main():
    22. scanner = BleakScanner()
    23. scanner.register_detection_callback(handle_advertisement)
    24. await scanner.start()
    25. try:
    26. while True:
    27. await asyncio.sleep(0.1) # Reduced sleep time for potentially more frequent updates
    28. except KeyboardInterrupt:
    29. await scanner.stop()
    30. if __name__ == "__main__":
    31. asyncio.run(main())

    This doesn't correctly parse my "presses" int, but for now i'm just looking at how fast it updates. This is what my script prints out:

    1. Puck
    2. Manufacturer Data: a4, Delta Time: 0.16 seconds
    3. Puck
    4. Manufacturer Data: a6, Delta Time: 0.00 seconds
    5. Puck
    6. Manufacturer Data: b6, Delta Time: 0.01 seconds
    7. Puck
    8. Manufacturer Data: b6, Delta Time: 0.28 seconds
    9. Puck
    10. Manufacturer Data: b8, Delta Time: 0.00 seconds
    11. Puck
    12. Manufacturer Data: c0, Delta Time: 0.01 seconds
    13. Puck
    14. Manufacturer Data: c6, Delta Time: 0.01 seconds
    15. Puck
    16. Manufacturer Data: c8, Delta Time: 0.00 seconds
    17. Puck
    18. Manufacturer Data: cc, Delta Time: 0.01 seconds
    19. Puck
    20. Manufacturer Data: c6, Delta Time: 0.30 seconds
    21. Puck
    22. Manufacturer Data: cc, Delta Time: 0.01 seconds
    23. Puck
    24. Manufacturer Data: ce, Delta Time: 0.00 seconds
    25. Puck
    26. Manufacturer Data: d0, Delta Time: 0.00 seconds
    27. Puck
    28. Manufacturer Data: f0, Delta Time: 0.00 seconds

    Ok - so that's mostly good. But here's the weird part. In mac OS terminal, I visually see the updates coming in roughly once a second, even though delta time is a fraction of that.

    Sooooo now i'm wondering if my mac BLE service is just buffering stuff up or something? If this is a dead end, I can go back to a direct blue tooth connection which definitely seems to be much faster. But wow it'd be cool if i could do this with advertising. :). Thanks!

  • in Puck.js, Pixl.js and MDBT42
    Avatar for DrewS

    Thanks so much for this great feedback. There's a lot for me to try here, but real quick on the battery - that seems to be good. It's at 75%.

    I'll start trying these wonderful suggestions and report back. Thx again for the super quick in depth responses!

  • in Puck.js, Pixl.js and MDBT42
    Avatar for DrewS

    I have a hobby project where my puck.js sends accel data:

    1. Puck.on("accel", function (a) {
    2. var accel = a.acc;
    3. accel.name = "acc";
    4. var gyro = a.gyro;
    5. gyro.name = "gyro";
    6. var magnitude = {}
    7. magnitude.name = "mag";
    8. console.log(JSON.stringify(accel));
    9. console.log(JSON.stringify(gyro));
    10. magnitude.value = Math.sqrt(accel.x * accel.x + accel.y * accel.y + accel.z * accel.z);
    11. console.log(JSON.stringify(magnitude));

    Then I have a python script that uses bleak to connect to the puck, and listen for that console output, which I then pass along to another toy program.

    1. print ("Connecting to puck " + puck_device.address)
    2. async with BleakClient(puck_device.address) as client:
    3. await client.start_notify(UUID_NORDIC_RX, uart_data_received)
    4. await asyncio.sleep(100000000.0) # Keep the connection alive
    5. await ws_server.wait_closed()

    This all works fine for a while, but i've noticed that after a few minutes the speed of the accel data i'm getting starts to slow down, and eventually the connection drops.

    Any suggestions on what might be happening? I looked into trying to do this by purely advertising the IMU data and not making an actual two way connection, but it doesn't look like I get a very high frequency signal that way. Just a few updates per second, and I need closer to 10 updates per second.

    Thanks! p.s. i'm new to BLE so might be doing this a bit backwards.

  • in Puck.js, Pixl.js and MDBT42
    Avatar for DrewS

    Ok, tried the uart:false flag but no luck there. I did look more deeply into the serial-console as well, but (perhaps not surprisingly) it's a bit of a stretch for my technical level. I'll see how far I can get using the BLE keyboard input module instead for now. Thx again for all the help.

    • 9 comments
    • 1.78k views
    @DrewS replied
  • in Puck.js, Pixl.js and MDBT42
    Avatar for DrewS

    Thanks Gordon - I've been away from my computer but will give these a shot and follow up later this week. Much appreciated!

  • in Puck.js, Pixl.js and MDBT42
    Avatar for DrewS

    Awesome tips -thx! Responses:

    Is OpenEmu running on the same Mac you're developing with?

    yup!

    Maybe try NRF.sendHIDReport([buttons, x, y], () => { inside a try { ... } catch...

    Done

    I'd remove the print statements.

    Ahh - didn't realize that could cause issues. Removed.

    You could add a little LED flash in your code so you can see if update is still executing correctly

    Done - great suggestion. I have it flashing blue when it starts the sendHID function, red when NRF.sendHIDReport() is successful, and green when it's sending a button press.

    With those changes it does expose something weird. When I connect via the IDE, I can flash the device and see both the red and blue LEDs blinking. When I disconnect from the IDE and only connect via the mac bluetooth menu, I only ever see the blue led blinking. So NRF.sendHIDReport() never seems to be successful, unless I connect via the IDE...

    Does that add up to anything obviously incorrect?

    1. //https://github.com/espruino/BangleApps/blob/master/apps/boot/hid_info.txt
    2. var joystick_report = new Uint8Array([
    3. 0x05, 0x01, // Usage Page (Generic Desktop)
    4. 0x09, 0x04, // Usage (Joystick)
    5. 0xA1, 0x01, // Collection (Application)
    6. 0x09, 0x01, // Usage (Pointer)
    7. 0xA1, 0x00, // Collection (Physical)
    8. // Buttons
    9. 0x05, 0x09, // Usage Page (Buttons)
    10. 0x19, 0x01, // Usage Minimum (1)
    11. 0x29, 0x05, // Usage Maximum (5)
    12. 0x15, 0x00, // Logical Minimum (0)
    13. 0x25, 0x01, // Logical Maximum (1)
    14. 0x95, 0x05, // Report Count (5)
    15. 0x75, 0x01, // Report Size (1)
    16. 0x81, 0x02, // Input (Data, Variable, Absolute)
    17. // padding bits
    18. 0x95, 0x03, // Report Count (3)
    19. 0x75, 0x01, // Report Size (1)
    20. 0x81, 0x03, // Input (Constant)
    21. // Stick
    22. 0x05, 0x01, // Usage Page (Generic Desktop)
    23. 0x09, 0x30, // Usage (X)
    24. 0x09, 0x31, // Usage (Y)
    25. 0x15, 0x81, // Logical Minimum (-127)
    26. 0x25, 0x7f, // Logical Maximum (127)
    27. 0x75, 0x08, // Report Size (8)
    28. 0x95, 0x02, // Report Count (2)
    29. 0x81, 0x02, // Input (Data, Variable, Absolute)
    30. 0xC0, // End Collection (Physical)
    31. 0xC0 // End Collection (Application)
    32. ]);
    33. NRF.setServices(undefined, { hid : joystick_report });
    34. Puck.accelOn(); // default is 12.5Hz, with gyro
    35. var sendInProgress = false; // Only send one message at a time, do not flood
    36. const sendHid = function (x, y, btn1, btn2, btn3, btn4, btn5, cb) {
    37. try {
    38. const buttons = (btn5<<4) | (btn4<<3) | (btn3<<2) | (btn2<<1) | (btn1<<0);
    39. if (!sendInProgress) {
    40. sendInProgress = true;
    41. try {
    42. NRF.sendHIDReport([buttons, x, y], () =>
    43. {
    44. sendInProgress = false;
    45. //print ([buttons, x, y]);
    46. digitalPulse(LED1,1,1);
    47. if (buttons == 1) {
    48. digitalPulse(LED2,1,1);
    49. }
    50. }
    51. );
    52. } catch(e) {
    53. print(e);
    54. }
    55. }
    56. } catch(e) {
    57. print(e);
    58. }
    59. };
    60. function update() {
    61. const btn1 = BTN1.read();
    62. const accel = Puck.accel();
    63. var x = accel.acc.x/128000;
    64. var y = accel.acc.y/128000;
    65. // rescale to approximately [-127:127]
    66. x *= 10;
    67. y *= 10;
    68. // I assume the joystick requires ints
    69. x = Math.floor( x * 127 );
    70. y = Math.floor( y * 127);
    71. // check limits
    72. if (x > 127) x = 127;
    73. else if (x < -127) x = -127;
    74. if (y > 127) y = 127;
    75. else if (y < -127) y = -127;
    76. sendHid(x & 0xff, y & 0xff, btn1, false, false, false, false);
    77. //print ("x", x, "y", y, "btn1", btn1);
    78. }
    79. setInterval(update, 100); // 10 Hz
  • in Puck.js, Pixl.js and MDBT42
    Avatar for DrewS

    Ok - I have some incremental progress. Below is my modified js for the puck. I'm testing it in OpenEmu on my Mac. The puck shows up as a joystick (sometimes as 'unknown' and sometimes as 'Puck.js 80b6'). Once I got Y axis analog input triggering, but no buttons and nothing on the x axis.

    I also still get the message "BLE Connected, queueing BLE restart for later" in the IDE, but I seem to always get that even when using the BLE Keyboard HID. I've pretty aggressively been disconnecting the puck from bluetooth after flashing it with my program, so I think the bluetooth stack is automatically restarting in HID mode (which seems to be reflected in the device showing up in OpenEMU's joystick list). That said, my workflow could still be wrong here.

    Anyway, let me know if you spot anything obviously incorrect. Thx again!

    1. //https://github.com/espruino/BangleApps/blob/master/apps/boot/hid_info.txt
    2. var joystick_report = new Uint8Array([
    3. 0x05, 0x01, // Usage Page (Generic Desktop)
    4. 0x09, 0x04, // Usage (Joystick)
    5. 0xA1, 0x01, // Collection (Application)
    6. 0x09, 0x01, // Usage (Pointer)
    7. 0xA1, 0x00, // Collection (Physical)
    8. // Buttons
    9. 0x05, 0x09, // Usage Page (Buttons)
    10. 0x19, 0x01, // Usage Minimum (1)
    11. 0x29, 0x05, // Usage Maximum (5)
    12. 0x15, 0x00, // Logical Minimum (0)
    13. 0x25, 0x01, // Logical Maximum (1)
    14. 0x95, 0x05, // Report Count (5)
    15. 0x75, 0x01, // Report Size (1)
    16. 0x81, 0x02, // Input (Data, Variable, Absolute)
    17. // padding bits
    18. 0x95, 0x03, // Report Count (3)
    19. 0x75, 0x01, // Report Size (1)
    20. 0x81, 0x03, // Input (Constant)
    21. // Stick
    22. 0x05, 0x01, // Usage Page (Generic Desktop)
    23. 0x09, 0x30, // Usage (X)
    24. 0x09, 0x31, // Usage (Y)
    25. 0x15, 0x81, // Logical Minimum (-127)
    26. 0x25, 0x7f, // Logical Maximum (127)
    27. 0x75, 0x08, // Report Size (8)
    28. 0x95, 0x02, // Report Count (2)
    29. 0x81, 0x02, // Input (Data, Variable, Absolute)
    30. 0xC0, // End Collection (Physical)
    31. 0xC0 // End Collection (Application)
    32. ]);
    33. NRF.setServices(undefined, { hid : joystick_report });
    34. Puck.accelOn(); // default is 12.5Hz, with gyro
    35. var sendInProgress = false; // Only send one message at a time, do not flood
    36. const sendHid = function (x, y, btn1, btn2, btn3, btn4, btn5, cb) {
    37. try {
    38. const buttons = (btn5<<4) | (btn4<<3) | (btn3<<2) | (btn2<<1) | (btn1<<0);
    39. if (!sendInProgress) {
    40. sendInProgress = true;
    41. print ([buttons, x, y]);
    42. NRF.sendHIDReport([buttons, x, y], () => {
    43. sendInProgress = false;
    44. if (cb) cb();
    45. });
    46. }
    47. } catch(e) {
    48. print(e);
    49. }
    50. };
    51. function update() {
    52. const btn1 = BTN1.read();
    53. const accel = Puck.accel();
    54. var x = accel.acc.x/128000;
    55. var y = accel.acc.y/128000;
    56. // rescale to approximately [-127:127]
    57. x *= 10;
    58. y *= 10;
    59. // I assume the joystick requires ints
    60. x = Math.floor( x * 127 );
    61. y = Math.floor( y * 127);
    62. // check limits
    63. if (x > 127) x = 127;
    64. else if (x < -127) x = -127;
    65. if (y > 127) y = 127;
    66. else if (y < -127) y = -127;
    67. sendHid(x & 0xff, y & 0xff, btn1, false, false, false, false);
    68. //print ("x", x, "y", y, "btn1", btn1);
    69. }
    70. setInterval(update, 100); // 10 Hz
Actions