From 4ccca0061241872940d6e85131ba190e9757a4e2 Mon Sep 17 00:00:00 2001 From: Philip Pitts <84428015+philippitts@users.noreply.github.com> Date: Thu, 21 Sep 2023 17:28:30 -0400 Subject: [PATCH] Support Ganglion v3 firmware detection --- .../java/openbci_gui_helpers/GUIHelper.java | 134 +++++++++--------- .../examples/TestDiscovery.java | 17 +-- .../examples/TestNativeDiscovery.java | 19 ++- modules/bglib/build.cmake | 2 + modules/bglib/src/callbacks.cpp | 16 ++- modules/bglib/src/openbci_gui_helpers.cpp | 9 +- modules/common/include/serialization.h | 16 +++ modules/common/src/serialization.cpp | 14 ++ modules/native-ble/build.cmake | 2 + .../src/openbci_gui_native_helpers.cpp | 51 +++++-- 10 files changed, 169 insertions(+), 111 deletions(-) create mode 100644 modules/common/include/serialization.h create mode 100644 modules/common/src/serialization.cpp diff --git a/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/GUIHelper.java b/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/GUIHelper.java index 60a31a1..29b3b70 100644 --- a/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/GUIHelper.java +++ b/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/GUIHelper.java @@ -1,87 +1,76 @@ package openbci_gui_helpers; -import java.io.BufferedReader; import java.io.File; import java.io.InputStream; -import java.io.InputStreamReader; -import java.lang.reflect.Type; -import java.nio.charset.StandardCharsets; import java.nio.file.Files; -import java.nio.file.Paths; -import java.nio.file.Path; -import java.util.Map; -import java.util.stream.Collectors; +import java.util.ArrayList; import org.apache.commons.lang3.SystemUtils; import com.google.gson.Gson; -import com.google.gson.reflect.TypeToken; import com.sun.jna.Library; import com.sun.jna.Native; -public class GUIHelper -{ - private interface DllInterface extends Library - { - int scan_for_ganglions (String serial_port, int timeout_sec, byte[] output, int[] output_len); +public class GUIHelper { + public class GanglionDevice { + public int firmware_version; + public String identifier; + public String mac_address; } - private interface DllNativeInterface extends Library - { - int scan_for_ganglions (int timeout_sec, byte[] output, int[] output_len); + private interface DllInterface extends Library { + int scan_for_ganglions(String serial_port, int timeout_sec, byte[] output, int[] output_len); + } + + private interface DllNativeInterface extends Library { + int scan_for_ganglions(int timeout_sec, byte[] output, int[] output_len); } private static DllInterface instance; private static DllNativeInterface instance_native; - private static final String VERSION = "2.0.1"; + private static final String VERSION = "3.0.0"; - static - { + static { System.out.println("OpenBCI_GUI_Helpers Version: " + VERSION); String lib_name = "libGanglionScan.so"; String lib_native_name = "libGanglionNativeScan.so"; - if (SystemUtils.IS_OS_WINDOWS) - { + if (SystemUtils.IS_OS_WINDOWS) { lib_name = "GanglionScan.dll"; lib_native_name = "GanglionNativeScan.dll"; - } else if (SystemUtils.IS_OS_MAC) - { + } else if (SystemUtils.IS_OS_MAC) { lib_name = "libGanglionScan.dylib"; lib_native_name = "libGanglionNativeScan.dylib"; } // need to extract libraries from jar - String lib_path = unpack_from_jar (lib_name); - String lib_native_path = unpack_from_jar (lib_native_name); + String lib_path = unpack_from_jar(lib_name); + String lib_native_path = unpack_from_jar(lib_native_name); - instance = (DllInterface) Native.load (lib_path, DllInterface.class); - instance_native = (DllNativeInterface) Native.load (lib_native_path, DllNativeInterface.class); + instance = (DllInterface) Native.load(lib_path, DllInterface.class); + instance_native = (DllNativeInterface) Native.load(lib_native_path, DllNativeInterface.class); } - private static String unpack_from_jar (String lib_name) - { + private static String unpack_from_jar(String lib_name) { File file = null; InputStream link = null; - try - { - file = new File (lib_name); - if (file.exists ()) - file.delete (); - link = (GUIHelper.class.getResourceAsStream (lib_name)); + try { + file = new File(lib_name); + if (file.exists()) + file.delete(); + link = (GUIHelper.class.getResourceAsStream(lib_name)); if (SystemUtils.IS_OS_MAC) { String jarPath = GUIHelper.class.getProtectionDomain().getCodeSource().getLocation().getPath(); File jarFile = new File(jarPath); String libPathString = jarFile.getParentFile().getAbsolutePath() + File.separator + lib_name; return libPathString; } else { - Files.copy (link, file.getAbsoluteFile ().toPath ()); + Files.copy(link, file.getAbsoluteFile().toPath()); } return file.getAbsolutePath(); - } catch (Exception io) - { - io.printStackTrace (); + } catch (Exception io) { + io.printStackTrace(); System.err.println("native library: " + lib_name + " is not found in jar file"); System.err.println("file absolute to path: " + file.getAbsoluteFile().toPath()); System.err.println("file get absolute path: " + file.getAbsolutePath()); @@ -89,40 +78,51 @@ private static String unpack_from_jar (String lib_name) } } - public static Map scan_for_ganglions (String port_name, int timeout_sec) throws GanglionError - { + public static GanglionDevice[] scan_for_ganglions(String port_name, int timeout_sec) throws GanglionError { int[] len = new int[1]; byte[] output_json = new byte[10240]; - int ec = instance.scan_for_ganglions (port_name, timeout_sec, output_json, len); - if (ec != GanglionExitCodes.STATUS_OK.get_code ()) - { - throw new GanglionError ("Error in scan for ganglions", ec); + + int ec = instance.scan_for_ganglions(port_name, timeout_sec, output_json, len); + if (ec != GanglionExitCodes.STATUS_OK.get_code()) { + throw new GanglionError("Error in scan for ganglions", ec); } - String json = new String (output_json, 0, len[0]); - Gson gson = new Gson (); - Type type = new TypeToken> () - { - }.getType (); - Map map = gson.fromJson (json, type); - return map; + + String json = new String(output_json, 0, len[0]); + Gson gson = new Gson(); + + GanglionDevice[] devices = gson.fromJson(json, GanglionDevice[].class); + + ArrayList uniqueDevices = new ArrayList(); + for (GanglionDevice device : devices) { + if (!uniqueDevices.stream().anyMatch(entry -> entry.mac_address.equals(device.mac_address))) { + uniqueDevices.add(device); + } + } + + return uniqueDevices.toArray(new GanglionDevice[uniqueDevices.size()]); } - public static Map scan_for_ganglions (int timeout_sec) throws GanglionError - { + public static GanglionDevice[] scan_for_ganglions(int timeout_sec) throws GanglionError { int[] len = new int[1]; byte[] output_json = new byte[10240]; - int ec = instance_native.scan_for_ganglions (timeout_sec, output_json, len); - if (ec != GanglionExitCodes.STATUS_OK.get_code ()) - { - throw new GanglionError ("Error in scan for ganglions", ec); + + int ec = instance_native.scan_for_ganglions(timeout_sec, output_json, len); + if (ec != GanglionExitCodes.STATUS_OK.get_code()) { + throw new GanglionError("Error in scan for ganglions", ec); } - String json = new String (output_json, 0, len[0]); - Gson gson = new Gson (); - Type type = new TypeToken> () - { - }.getType (); - Map map = gson.fromJson (json, type); - return map; - } + String json = new String(output_json, 0, len[0]); + Gson gson = new Gson(); + + GanglionDevice[] devices = gson.fromJson(json, GanglionDevice[].class); + + ArrayList uniqueDevices = new ArrayList(); + for (GanglionDevice device : devices) { + if (!uniqueDevices.stream().anyMatch(entry -> entry.mac_address.equals(device.mac_address))) { + uniqueDevices.add(device); + } + } + + return uniqueDevices.toArray(new GanglionDevice[uniqueDevices.size()]); + } } diff --git a/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/examples/TestDiscovery.java b/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/examples/TestDiscovery.java index 543b26d..0aa598a 100644 --- a/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/examples/TestDiscovery.java +++ b/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/examples/TestDiscovery.java @@ -1,19 +1,16 @@ package openbci_gui_helpers.examples; -import java.util.Map; - import openbci_gui_helpers.GUIHelper; import openbci_gui_helpers.GanglionError; +import openbci_gui_helpers.GUIHelper.GanglionDevice; -public class TestDiscovery -{ +public class TestDiscovery { - public static void main (String[] args) throws GanglionError - { - Map map = GUIHelper.scan_for_ganglions (args[0], 3); - for (Map.Entry entry : map.entrySet ()) - { - System.out.println ("Key = " + entry.getKey () + ", Value = " + entry.getValue ()); + public static void main(String[] args) throws GanglionError { + GanglionDevice[] devices = GUIHelper.scan_for_ganglions(args[0], 3); + for (GanglionDevice device : devices) { + System.out.println("Identifier = " + device.identifier + ", Mac Address = " + device.mac_address + + ", Firmware = " + device.firmware_version); } } } diff --git a/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/examples/TestNativeDiscovery.java b/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/examples/TestNativeDiscovery.java index ca8dc7f..7aef12a 100644 --- a/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/examples/TestNativeDiscovery.java +++ b/java-package/openbci_gui_helpers/src/main/java/openbci_gui_helpers/examples/TestNativeDiscovery.java @@ -1,20 +1,17 @@ package openbci_gui_helpers.examples; -import java.util.Map; - import openbci_gui_helpers.GUIHelper; +import openbci_gui_helpers.GUIHelper.GanglionDevice; import openbci_gui_helpers.GanglionError; -public class TestNativeDiscovery -{ +public class TestNativeDiscovery { - public static void main (String[] args) throws GanglionError - { - Map map = GUIHelper.scan_for_ganglions (3); - for (Map.Entry entry : map.entrySet ()) - { - System.out.println ("Key = " + entry.getKey () + ", Value = " + entry.getValue ()); + public static void main(String[] args) throws GanglionError { + GanglionDevice[] devices = GUIHelper.scan_for_ganglions(3); + for (GanglionDevice device : devices) { + System.out.println("Identifier = " + device.identifier + ", Mac Address = " + device.mac_address + + ", Firmware = " + device.firmware_version); } - System.out.println ("Completed"); + System.out.println("Completed"); } } diff --git a/modules/bglib/build.cmake b/modules/bglib/build.cmake index f414a88..015f69f 100644 --- a/modules/bglib/build.cmake +++ b/modules/bglib/build.cmake @@ -2,6 +2,7 @@ SET(GANGLION_LIB "GanglionScan") add_library( ${GANGLION_LIB} SHARED + modules/common/src/serialization.cpp modules/bglib/src/callbacks.cpp modules/bglib/src/cmd_def.cpp modules/bglib/src/stubs.cpp @@ -13,6 +14,7 @@ target_include_directories( ${GANGLION_LIB} PUBLIC 3rdparty/json modules/bglib/include + modules/common/include ) set_property(TARGET ${GANGLION_LIB} PROPERTY POSITION_INDEPENDENT_CODE ON) diff --git a/modules/bglib/src/callbacks.cpp b/modules/bglib/src/callbacks.cpp index 85cc0ce..fb423c7 100644 --- a/modules/bglib/src/callbacks.cpp +++ b/modules/bglib/src/callbacks.cpp @@ -1,19 +1,21 @@ #include #include -#include +#include #include #include #include #include + #include "cmd_def.h" #include "openbci_gui_helpers.h" +#include "serialization.h" namespace GanglionDetails { extern int exit_code; - extern std::map devices; + extern std::list devices; } void ble_evt_gap_scan_response (const struct ble_msg_gap_scan_response_evt_t *msg) @@ -57,7 +59,15 @@ void ble_evt_gap_scan_response (const struct ble_msg_gap_scan_response_evt_t *ms mac_addr << ":"; } } - GanglionDetails::devices[std::string (name)] = mac_addr.str (); + + uint8_t firmware = strstr (name, "anglion 1.3") == NULL ? 2 : 3; + + GanglionDevice device; + device.identifier = name; + device.mac_address = mac_addr.str (); + device.firmware_version = firmware; + + GanglionDetails::devices.push_back (device); GanglionDetails::exit_code = (int)GanglionDetails::GanglionScanExitCodes::STATUS_OK; } } diff --git a/modules/bglib/src/openbci_gui_helpers.cpp b/modules/bglib/src/openbci_gui_helpers.cpp index fbe798d..ad53df8 100644 --- a/modules/bglib/src/openbci_gui_helpers.cpp +++ b/modules/bglib/src/openbci_gui_helpers.cpp @@ -1,12 +1,13 @@ #include #include -#include +#include #include #include #include #include "cmd_def.h" #include "openbci_gui_helpers.h" +#include "serialization.h" #include "uart.h" #ifdef _WIN32 @@ -15,14 +16,12 @@ #include #endif -#include "json.hpp" - using json = nlohmann::json; namespace GanglionDetails { int exit_code = (int)GanglionScanExitCodes::STATUS_OK; - std::map devices; + std::list devices; void output (uint8 len1, uint8 *data1, uint16 len2, uint8 *data2) { @@ -138,7 +137,7 @@ int scan_for_ganglions (char *serial_port, int timeout_sec, char *output_json, i json result (GanglionDetails::devices); std::string s = result.dump (); strcpy (output_json, s.c_str ()); - *output_len = s.length (); + *output_len = (int)s.length (); } ble_cmd_gap_end_procedure (); uart_close (); diff --git a/modules/common/include/serialization.h b/modules/common/include/serialization.h new file mode 100644 index 0000000..ef1345d --- /dev/null +++ b/modules/common/include/serialization.h @@ -0,0 +1,16 @@ +#pragma once + +#include "json.hpp" + +#include +#include + +typedef struct GanglionDevice +{ + std::string identifier; + std::string mac_address; + uint8_t firmware_version; +} GanglionDevice; + +void to_json (nlohmann::json &result, const GanglionDevice &device); +void from_json (const nlohmann::json &string, GanglionDevice &device); \ No newline at end of file diff --git a/modules/common/src/serialization.cpp b/modules/common/src/serialization.cpp new file mode 100644 index 0000000..e080249 --- /dev/null +++ b/modules/common/src/serialization.cpp @@ -0,0 +1,14 @@ +#include "serialization.h" + +void to_json (nlohmann::json &result, const GanglionDevice &device) +{ + result = nlohmann::json {{"identifier", device.identifier}, {"mac_address", device.mac_address}, + {"firmware_version", std::to_string (device.firmware_version)}}; +} + +void from_json (const nlohmann::json &string, GanglionDevice &device) +{ + string.at ("identifier").get_to (device.identifier); + string.at ("mac_address").get_to (device.mac_address); + string.at ("firmware_version").get_to (device.firmware_version); +} \ No newline at end of file diff --git a/modules/native-ble/build.cmake b/modules/native-ble/build.cmake index 60c65f8..b40dea6 100644 --- a/modules/native-ble/build.cmake +++ b/modules/native-ble/build.cmake @@ -2,12 +2,14 @@ SET(OPENBCI_GANGLION_NATIVE_LIB "GanglionNativeScan") add_library( ${OPENBCI_GANGLION_NATIVE_LIB} SHARED + modules/common/src/serialization.cpp modules/native-ble/src/openbci_gui_native_helpers.cpp ) target_include_directories( ${OPENBCI_GANGLION_NATIVE_LIB} PUBLIC modules/native-ble/include + modules/common/include 3rdparty/json 3rdparty/SimpleBLE/simpleble/include/simpleble ) diff --git a/modules/native-ble/src/openbci_gui_native_helpers.cpp b/modules/native-ble/src/openbci_gui_native_helpers.cpp index dbaaba2..fb86779 100644 --- a/modules/native-ble/src/openbci_gui_native_helpers.cpp +++ b/modules/native-ble/src/openbci_gui_native_helpers.cpp @@ -1,21 +1,24 @@ -#include #include -#include #include #include #include +#include + + +#include #include "SimpleBLE.h" #include "openbci_gui_native_helpers.h" - -#include "json.hpp" +#include "serialization.h" using json = nlohmann::json; +#define SOFTWARE_REVISION_CHARACTERISTIC "00002a28-0000-1000-8000-00805f9b34fb" +#define DEVICE_INFORMATION_SERVICE "0000180a-0000-1000-8000-00805f9b34fb" int scan_for_ganglions (int timeout_sec, char *output_json, int *output_len) { - std::map devices; + std::list peripherals; auto adapter_list = SimpleBLE::Adapter::get_adapters (); if (adapter_list.empty ()) { @@ -23,26 +26,44 @@ int scan_for_ganglions (int timeout_sec, char *output_json, int *output_len) } adapter_list[0].set_callback_on_scan_found ( - [&devices] (SimpleBLE::Peripheral peripheral) + [&peripherals] (SimpleBLE::Peripheral peripheral) { std::string identifier = peripheral.identifier (); - std::string mac_address = peripheral.address (); - if (strncmp (identifier.c_str (), "Ganglion", 8) == 0) - { - devices[identifier] = mac_address; - } - else if (strncmp (identifier.c_str (), "Simblee", 7) == 0) + + if (strncmp (identifier.c_str (), "Ganglion", 8) == 0 || + strncmp (identifier.c_str (), "Simblee", 7) == 0) { - devices[identifier] = mac_address; + peripherals.push_back (peripheral); } }); adapter_list[0].scan_for (timeout_sec * 1000); - json result (devices); + std::list devices; + for (auto peripheral : peripherals) + { + GanglionDevice device; + device.identifier = peripheral.identifier (); + device.mac_address = peripheral.address (); + + peripheral.connect (); + + // Read the software revision characteristic to get the firmware version + SimpleBLE::ByteArray software_revision = + peripheral.read (DEVICE_INFORMATION_SERVICE, SOFTWARE_REVISION_CHARACTERISTIC); + + peripheral.disconnect (); + + // Version should be in the form x.x.x stored as ASCII values in the data buffer + device.firmware_version = (software_revision[0] == '3') ? 3 : 2; + + devices.push_back (device); + } + + json result = devices; std::string s = result.dump (); strcpy (output_json, s.c_str ()); - *output_len = s.length (); + *output_len = (int)s.length (); return (int)GanglionScanExitCodes::STATUS_OK; }