Huami: Distinguish between alarm and workout activity type responses

This commit is contained in:
José Rebelo 2022-05-15 17:52:01 +01:00 committed by Gitea
parent 3b348a5d5d
commit f08fe7f65f
2 changed files with 39 additions and 18 deletions

View File

@ -165,10 +165,16 @@ public class HuamiService {
public static final byte[] COMMAND_ENABLE_DISCONNECT_NOTIFCATION = new byte[]{ENDPOINT_DISPLAY, 0x0c, 0x00, 0x01, 0, 0, 0, 0}; public static final byte[] COMMAND_ENABLE_DISCONNECT_NOTIFCATION = new byte[]{ENDPOINT_DISPLAY, 0x0c, 0x00, 0x01, 0, 0, 0, 0};
public static final byte[] COMMAND_DISABLE_DISCONNECT_NOTIFCATION = new byte[]{ENDPOINT_DISPLAY, 0x0c, 0x00, 0x00, 0, 0, 0, 0}; public static final byte[] COMMAND_DISABLE_DISCONNECT_NOTIFCATION = new byte[]{ENDPOINT_DISPLAY, 0x0c, 0x00, 0x00, 0, 0, 0, 0};
public static final byte[] COMMAND_REQUEST_ALARMS = new byte[]{0x0d}; public static final byte COMMAND_ALARMS = 0x0d;
public static final byte[] COMMAND_REQUEST_ALARMS_WITH_TIMES = new byte[]{(byte) 0xff,0x01,0x00,0x00,0x00}; public static final byte COMMAND_ALARMS_WITH_TIMES = 0x01;
public static final byte COMMAND_WORKOUT_ACTIVITY_TYPES = 0x11;
public static final byte COMMAND_GPS_VERSION = 0x0e;
public static final byte[] COMMAND_REQUEST_GPS_VERSION = new byte[]{0x0e}; public static final byte[] COMMAND_REQUEST_ALARMS = new byte[]{COMMAND_ALARMS};
public static final byte[] COMMAND_REQUEST_ALARMS_WITH_TIMES = new byte[]{(byte) 0xff, COMMAND_ALARMS_WITH_TIMES, 0x00, 0x00, 0x00};
public static final byte[] COMMAND_REQUEST_WORKOUT_ACTIVITY_TYPES = new byte[]{(byte) 0xff, COMMAND_WORKOUT_ACTIVITY_TYPES, 0x00, 0x00, 0x00};
public static final byte[] COMMAND_REQUEST_GPS_VERSION = new byte[]{COMMAND_GPS_VERSION};
// The third byte controls the threshold, in minutes // The third byte controls the threshold, in minutes
// The last 8 bytes represent 2 separate time intervals for the inactivity warnings // The last 8 bytes represent 2 separate time intervals for the inactivity warnings

View File

@ -225,6 +225,10 @@ import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiConst.PREF
import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiConst.PREF_HUAMI_VIBRATION_TRY_INCOMING_CALL; import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiConst.PREF_HUAMI_VIBRATION_TRY_INCOMING_CALL;
import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiConst.PREF_HUAMI_VIBRATION_TRY_INCOMING_SMS; import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiConst.PREF_HUAMI_VIBRATION_TRY_INCOMING_SMS;
import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiConst.PREF_HUAMI_VIBRATION_TRY_PREFIX; import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiConst.PREF_HUAMI_VIBRATION_TRY_PREFIX;
import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService.COMMAND_ALARMS;
import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService.COMMAND_ALARMS_WITH_TIMES;
import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService.COMMAND_GPS_VERSION;
import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService.COMMAND_WORKOUT_ACTIVITY_TYPES;
import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService.DISPLAY_ITEM_BIT_CLOCK; import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService.DISPLAY_ITEM_BIT_CLOCK;
import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService.ENDPOINT_DISPLAY_ITEMS; import static nodomain.freeyourgadget.gadgetbridge.devices.huami.HuamiService.ENDPOINT_DISPLAY_ITEMS;
import static nodomain.freeyourgadget.gadgetbridge.devices.miband.MiBandConst.DEFAULT_VALUE_VIBRATION_COUNT; import static nodomain.freeyourgadget.gadgetbridge.devices.miband.MiBandConst.DEFAULT_VALUE_VIBRATION_COUNT;
@ -2043,18 +2047,19 @@ public class HuamiSupport extends AbstractBTLEDeviceSupport {
} }
} }
byte[] alarmConfigurationReassemblyBuffer; byte[] reassemblyBuffer;
byte reassemblyType = 0x00;
private void handleConfigurationInfo(byte[] value) { private void handleConfigurationInfo(byte[] value) {
if (value == null || value.length < 4) { if (value == null || value.length < 4) {
return; return;
} }
if (value[0] == 0x10 && value[2] == 0x01) { if (value[0] == 0x10 && value[2] == 0x01) {
if (value[1] == 0x0e) { if (value[1] == COMMAND_GPS_VERSION) {
String gpsVersion = new String(value, 3, value.length - 3); String gpsVersion = new String(value, 3, value.length - 3);
LOG.info("got gps version = " + gpsVersion); LOG.info("got gps version = " + gpsVersion);
gbDevice.setFirmwareVersion2(gpsVersion); gbDevice.setFirmwareVersion2(gpsVersion);
} else if (value[1] == 0x0d) { } else if (value[1] == COMMAND_ALARMS) {
LOG.info("got alarms from watch"); LOG.info("got alarms from watch");
decodeAndUpdateAlarmStatus(value, false); decodeAndUpdateAlarmStatus(value, false);
} else { } else {
@ -2062,26 +2067,38 @@ public class HuamiSupport extends AbstractBTLEDeviceSupport {
} }
} else if (value[0] == ((byte) 0x80) && value[1] == 0x01) { } else if (value[0] == ((byte) 0x80) && value[1] == 0x01) {
boolean done = false; boolean done = false;
if (value[2] == 0x00 || value[2] == (byte) 0xc0 && (value[4] == 0x01 && value[5] == 0x00 && value[6] == 0x00 && value[7] == 0x00 && value[8] == 0x01)) { // first chunk or complete data if (value[2] == 0x00 || value[2] == (byte) 0xc0) { // first chunk or complete data
alarmConfigurationReassemblyBuffer = new byte[value.length - 8]; reassemblyBuffer = new byte[value.length - 8];
System.arraycopy(value, 8, alarmConfigurationReassemblyBuffer, 0, alarmConfigurationReassemblyBuffer.length); reassemblyType = value[4];
System.arraycopy(value, 8, reassemblyBuffer, 0, reassemblyBuffer.length);
if (value[2] == (byte) 0xc0) { if (value[2] == (byte) 0xc0) {
done = true; done = true;
} }
} else if (alarmConfigurationReassemblyBuffer != null && (value[2] == 0x40 || value[2] == (byte) 0x80)) { } else if (reassemblyBuffer != null && (value[2] == 0x40 || value[2] == (byte) 0x80)) {
byte[] payload = new byte[value.length - 4]; byte[] payload = new byte[value.length - 4];
System.arraycopy(value, 4, payload, 0, payload.length); System.arraycopy(value, 4, payload, 0, payload.length);
alarmConfigurationReassemblyBuffer = ArrayUtils.addAll(alarmConfigurationReassemblyBuffer, payload); reassemblyBuffer = ArrayUtils.addAll(reassemblyBuffer, payload);
if (value[2] == (byte) 0x80) { if (value[2] == (byte) 0x80) {
done = true; done = true;
} }
} }
if (!done) { if (!done) {
LOG.info("got chunk of alarm configuration data"); LOG.info("got chunk of configuration data for {}", String.format("0x%x", reassemblyType));
} else { } else {
LOG.info("got full/reassembled configuration data"); LOG.info("got full/reassembled configuration data");
decodeAndUpdateAlarmStatus(alarmConfigurationReassemblyBuffer, true); switch (reassemblyType) {
alarmConfigurationReassemblyBuffer = null; case COMMAND_ALARMS_WITH_TIMES:
decodeAndUpdateAlarmStatus(reassemblyBuffer, true);
break;
case COMMAND_WORKOUT_ACTIVITY_TYPES:
LOG.warn("got workout activity types, not handled");
break;
default:
LOG.warn("got unknown chunked configuration response for {}, not handled", String.format("0x%x", reassemblyType));
break;
}
reassemblyBuffer = null;
} }
} else { } else {
LOG.warn("unknown response got from configuration request " + GB.hexdump(value, 0, -1)); LOG.warn("unknown response got from configuration request " + GB.hexdump(value, 0, -1));
@ -2535,10 +2552,8 @@ public class HuamiSupport extends AbstractBTLEDeviceSupport {
public void onTestNewFunction() { public void onTestNewFunction() {
//requestMTU(23); //requestMTU(23);
try { try {
final TransactionBuilder builder = performInitialized("test pattern"); final TransactionBuilder builder = performInitialized("test request");
final VibrationProfile profile = VibrationProfile.getProfile(VibrationProfile.ID_SHORT, (short) 2); writeToConfiguration(builder, HuamiService.COMMAND_REQUEST_WORKOUT_ACTIVITY_TYPES);
setVibrationPattern(builder, HuamiVibrationPatternNotificationType.APP_ALERTS, true, profile);
builder.queue(getQueue()); builder.queue(getQueue());
} catch (final Exception e) { } catch (final Exception e) {
LOG.error("onTestNewFunction failed", e); LOG.error("onTestNewFunction failed", e);