1
0
mirror of https://codeberg.org/Freeyourgadget/Gadgetbridge synced 2024-11-28 12:56:49 +01:00

SuperCars: switch from data replay to generated data

- use encryption to create data rather then replay captured BLE traffic
- use periodical data sender, as is required by the BLE module
- extract string resources
This commit is contained in:
vanous 2022-10-02 08:47:08 +02:00
parent 5f7674fe39
commit 51b7f28a8b
6 changed files with 211 additions and 175 deletions

View File

@ -2,6 +2,7 @@ package nodomain.freeyourgadget.gadgetbridge.devices.supercars;
import android.content.Intent; import android.content.Intent;
import android.os.Bundle; import android.os.Bundle;
import android.os.CountDownTimer;
import android.widget.CheckBox; import android.widget.CheckBox;
import android.widget.CompoundButton; import android.widget.CompoundButton;
@ -19,6 +20,12 @@ public class ControlActivity extends AbstractGBActivity implements JoystickView.
LocalBroadcastManager localBroadcastManager; LocalBroadcastManager localBroadcastManager;
boolean lights = false; boolean lights = false;
boolean turbo = false; boolean turbo = false;
CountDownTimer periodicDataSenderRunner;
SuperCarsConstants.Direction direction = SuperCarsConstants.Direction.CENTER;
SuperCarsConstants.Movement movement = SuperCarsConstants.Movement.IDLE;
SuperCarsConstants.Speed speed = SuperCarsConstants.Speed.NORMAL;
SuperCarsConstants.Light light = SuperCarsConstants.Light.OFF;
@Override @Override
protected void onCreate(Bundle savedInstanceState) { protected void onCreate(Bundle savedInstanceState) {
@ -41,52 +48,84 @@ public class ControlActivity extends AbstractGBActivity implements JoystickView.
lights = isChecked; lights = isChecked;
} }
}); });
//when this activity is open, data is sent continuously every 200ms
periodicDataSender();
periodicDataSenderRunner.start();
} }
private void sendLocalBroadcast(Intent intent) { private void sendLocalBroadcast(Intent intent) {
localBroadcastManager.sendBroadcast(intent); localBroadcastManager.sendBroadcast(intent);
} }
public void periodicDataSender() {
periodicDataSenderRunner = new CountDownTimer(Long.MAX_VALUE, 200) {
public void onTick(long millisUntilFinished) {
create_intent_with_data();
}
public void onFinish() {
start();
}
};
}
private void create_intent_with_data() {
Intent intent = new Intent(SuperCarsSupport.COMMAND_DRIVE_CONTROL);
intent.putExtra(SuperCarsSupport.EXTRA_DIRECTION, direction);
intent.putExtra(SuperCarsSupport.EXTRA_MOVEMENT, movement);
intent.putExtra(SuperCarsSupport.EXTRA_SPEED, speed);
intent.putExtra(SuperCarsSupport.EXTRA_LIGHT, light);
sendLocalBroadcast(intent);
}
@Override @Override
public void onJoystickMoved(float xPercent, float yPercent, int id) { public void onJoystickMoved(float xPercent, float yPercent, int id) {
if (yPercent != 0 && yPercent != 0) {
SuperCarsConstants.Directions command;
SuperCarsConstants.SpeedModes mode;
if (yPercent < 0) { if (yPercent < 0) {
command = SuperCarsConstants.Directions.UP; movement = SuperCarsConstants.Movement.UP;
if (xPercent < -0.5) { } else if (yPercent > 0) {
command = SuperCarsConstants.Directions.UP_LEFT; movement = SuperCarsConstants.Movement.DOWN;
} else if (xPercent > 0.5) { } else {
command = SuperCarsConstants.Directions.UP_RIGHT; movement = SuperCarsConstants.Movement.IDLE;
}
} else {
command = SuperCarsConstants.Directions.DOWN;
if (xPercent < -0.5) {
command = SuperCarsConstants.Directions.DOWN_LEFT;
} else if (xPercent > 0.5) {
command = SuperCarsConstants.Directions.DOWN_RIGHT;
}
}
mode = SuperCarsConstants.SpeedModes.NORMAL;
if (lights) {
mode = SuperCarsConstants.SpeedModes.LIGHTS;
if (turbo) {
mode = SuperCarsConstants.SpeedModes.TURBO_LIGHTS;
}
} else {
if (turbo) {
mode = SuperCarsConstants.SpeedModes.TURBO;
}
}
Intent intent = new Intent(SuperCarsSupport.COMMAND_DRIVE_CONTROL);
intent.putExtra(SuperCarsSupport.EXTRA_DIRECTION, command);
intent.putExtra(SuperCarsSupport.EXTRA_MODE, mode);
sendLocalBroadcast(intent);
} }
if (xPercent < -0.5) {
direction = SuperCarsConstants.Direction.LEFT;
} else if (xPercent > 0.5) {
direction = SuperCarsConstants.Direction.RIGHT;
} else {
direction = SuperCarsConstants.Direction.CENTER;
}
if (lights) {
light = SuperCarsConstants.Light.ON;
} else {
light = SuperCarsConstants.Light.OFF;
}
if (turbo) {
speed = SuperCarsConstants.Speed.TURBO;
} else {
speed = SuperCarsConstants.Speed.NORMAL;
}
}
@Override
protected void onDestroy() {
super.onDestroy();
periodicDataSenderRunner.cancel();
}
@Override
protected void onPostResume() {
super.onPostResume();
periodicDataSenderRunner.start();
}
@Override
protected void onPause() {
super.onPause();
periodicDataSenderRunner.cancel();
} }
} }

View File

@ -1,6 +1,7 @@
//GPL-3.0 license /* Original code taken from https://github.com/efficientisoceles/JoystickView
//https://github.com/efficientisoceles/JoystickView Licenced under the terms of the GNU General Public License version 3
Copyright (C) Daniel Su <goldenember@gmail.com>
*/
package nodomain.freeyourgadget.gadgetbridge.devices.supercars; package nodomain.freeyourgadget.gadgetbridge.devices.supercars;
import android.content.Context; import android.content.Context;
@ -109,9 +110,12 @@ public class JoystickView extends SurfaceView implements SurfaceHolder.Callback,
drawJoystick(constrainedX, constrainedY); drawJoystick(constrainedX, constrainedY);
joystickCallback.onJoystickMoved((constrainedX - centerX) / baseRadius, (constrainedY - centerY) / baseRadius, getId()); joystickCallback.onJoystickMoved((constrainedX - centerX) / baseRadius, (constrainedY - centerY) / baseRadius, getId());
} }
} else if (e.getAction() == MotionEvent.ACTION_UP) {
drawJoystick(centerX, centerY);
joystickCallback.onJoystickMoved(0, 0, getId());
} else } else
drawJoystick(centerX, centerY); drawJoystick(centerX, centerY);
joystickCallback.onJoystickMoved(0, 0, getId()); //joystickCallback.onJoystickMoved(0, 0, getId());
} }
return true; return true;
} }

View File

@ -4,6 +4,7 @@ import java.util.UUID;
public class SuperCarsConstants { public class SuperCarsConstants {
//https://gist.github.com/scrool/e79d6a4cb50c26499746f4fe473b3768#encryption
public static final byte[] aes_key = new byte[]{(byte) 0x34, (byte) 0x52, (byte) 0x2A, (byte) 0x5B, (byte) 0x7A, (byte) 0x6E, (byte) 0x49, (byte) 0x2C, (byte) 0x08, (byte) 0x09, (byte) 0x0A, (byte) 0x9D, (byte) 0x8D, (byte) 0x2A, (byte) 0x23, (byte) 0xF8}; public static final byte[] aes_key = new byte[]{(byte) 0x34, (byte) 0x52, (byte) 0x2A, (byte) 0x5B, (byte) 0x7A, (byte) 0x6E, (byte) 0x49, (byte) 0x2C, (byte) 0x08, (byte) 0x09, (byte) 0x0A, (byte) 0x9D, (byte) 0x8D, (byte) 0x2A, (byte) 0x23, (byte) 0xF8};
public static final UUID SERVICE_UUID_FFF = UUID.fromString("0000fff0-0000-1000-8000-00805f9b34fb"); public static final UUID SERVICE_UUID_FFF = UUID.fromString("0000fff0-0000-1000-8000-00805f9b34fb");
@ -16,118 +17,20 @@ public class SuperCarsConstants {
public static final UUID CHARACTERISTIC_UUID_FD1 = UUID.fromString("0000fd01-0000-1000-8000-00805f9b34fb"); public static final UUID CHARACTERISTIC_UUID_FD1 = UUID.fromString("0000fd01-0000-1000-8000-00805f9b34fb");
public static final UUID CHARACTERISTIC_UUID_FD2 = UUID.fromString("0000fd02-0000-1000-8000-00805f9b34fb"); public static final UUID CHARACTERISTIC_UUID_FD2 = UUID.fromString("0000fd02-0000-1000-8000-00805f9b34fb");
public static final byte[] idle_normal = new byte[]{0x02, 0x5e, 0x69, 0x5a, 0x48, (byte) 0xff, 0x2a, 0x43, (byte) 0x8c, (byte) 0xa6, (byte) 0x80, (byte) 0xf8, 0x3e, 0x04, (byte) 0xe4, 0x5d}; public enum Speed {
public static final byte[] up_normal = new byte[]{0x29, 0x60, (byte) 0x9c, 0x66, 0x48, 0x52, (byte) 0xcf, (byte) 0xf1, (byte) 0xb0, (byte) 0xf0, (byte) 0xcb, (byte) 0xb9, (byte) 0x80, 0x14, (byte) 0xbd, 0x2c}; NORMAL, TURBO
public static final byte[] down_normal = new byte[]{0x03, 0x20, (byte) 0x99, 0x09, (byte) 0xba, (byte) 0x9d, (byte) 0xa1, (byte) 0xc8, (byte) 0xb9, (byte) 0x86, 0x16, 0x3c, 0x6d, 0x48, 0x46, 0x55};
public static final byte[] up_left_normal = new byte[]{(byte) 0x99, 0x28, (byte) 0xe5, (byte) 0x90, (byte) 0xdf, (byte) 0xe8, 0x21, 0x48, 0x5f, 0x41, 0x4f, (byte) 0xbb, 0x63, 0x3d, 0x5c, 0x4e};
public static final byte[] up_right_normal = new byte[]{0x0f, 0x2c, (byte) 0xe5, 0x66, 0x62, (byte) 0xd4, (byte) 0xfd, (byte) 0x9d, 0x32, (byte) 0xa4, 0x4f, 0x10, 0x2b, (byte) 0xf2, 0x0a, (byte) 0xa7};
public static final byte[] down_left_normal = new byte[]{(byte) 0x98, (byte) 0xce, (byte) 0x98, 0x1d, 0x58, (byte) 0xd1, 0x15, (byte) 0xaf, (byte) 0xe1, 0x19, 0x60, (byte) 0xbf, 0x46, 0x13, (byte) 0x92, 0x5c};
public static final byte[] down_right_normal = new byte[]{(byte) 0xf2, 0x52, 0x0f, (byte) 0xba, 0x31, 0x44, (byte) 0xfb, 0x11, 0x46, (byte) 0x8f, (byte) 0xe0, (byte) 0x80, (byte) 0xc6, (byte) 0xc2, (byte) 0xc2, 0x3c};
public static final byte[] idle_lights = new byte[]{0x39, (byte) 0xb5, 0x3b, (byte) 0x9b, (byte) 0xb7, (byte) 0xa0, (byte) 0xe0, (byte) 0xd6, 0x52, 0x54, (byte) 0xf9, (byte) 0xea, (byte) 0x84, 0x5d, (byte) 0xde, (byte) 0xee};
public static final byte[] up_lights = new byte[]{0x2e, (byte) 0x90, (byte) 0xb1, (byte) 0xd3, 0x6b, (byte) 0x8f, (byte) 0xad, 0x5f, (byte) 0x96, 0x7d, (byte) 0xb3, 0x2e, 0x6e, 0x6c, (byte) 0xfd, 0x6e};
public static final byte[] down_lights = new byte[]{0x2b, 0x41, 0x0a, 0x47, (byte) 0xce, 0x03, 0x59, (byte) 0xe0, 0x4d, 0x75, (byte) 0xfd, 0x0e, (byte) 0xe3, (byte) 0x9f, (byte) 0xe1, (byte) 0xbd};
public static final byte[] up_right_lights = new byte[]{0x06, 0x4e, 0x40, (byte) 0x8c, 0x32, 0x52, (byte) 0xfb, 0x32, 0x7b, (byte) 0xd9, (byte) 0xfc, 0x54, (byte) 0x9b, 0x53, (byte) 0xee, 0x3e};
public static final byte[] up_left_lights = new byte[]{0x23, 0x77, 0x1d, (byte) 0xc2, 0x2a, 0x73, (byte) 0x89, (byte) 0x99, 0x2f, 0x53, (byte) 0xac, 0x59, 0x38, (byte) 0xc1, 0x78, (byte) 0x91};
public static final byte[] down_right_lights = new byte[]{(byte) 0xbb, (byte) 0xd6, 0x2a, (byte) 0xac, 0x32, (byte) 0x8c, (byte) 0x9e, 0x31, 0x65, 0x33, (byte) 0xc8, 0x0e, (byte) 0x9a, (byte) 0xcb, (byte) 0xf6, 0x4b};
public static final byte[] down_left_lights = new byte[]{0x6e, (byte) 0xaa, (byte) 0xf0, (byte) 0xc0, (byte) 0x85, (byte) 0x8f, 0x14, 0x77, 0x6f, (byte) 0xd8, (byte) 0xf0, 0x71, 0x39, (byte) 0xa0, 0x08, (byte) 0xf2};
public static final byte[] idle_turbo = new byte[]{(byte) 0xd9, (byte) 0x94, (byte) 0xb3, (byte) 0x71, (byte) 0xc3, (byte) 0xbe, (byte) 0x2a, (byte) 0x9a, (byte) 0x9d, (byte) 0x04, (byte) 0x88, (byte) 0xa1, (byte) 0x04, (byte) 0x4b, (byte) 0x7f, (byte) 0x67};
public static final byte[] up_turbo = new byte[]{(byte) 0xe6, (byte) 0x55, (byte) 0x67, (byte) 0xda, (byte) 0x8e, (byte) 0x6c, (byte) 0x56, (byte) 0x0d, (byte) 0x09, (byte) 0xd3, (byte) 0x73, (byte) 0x3a, (byte) 0x7f, (byte) 0x47, (byte) 0xff, (byte) 0x06};
public static final byte[] down_turbo = new byte[]{(byte) 0xce, (byte) 0xc2, (byte) 0xff, (byte) 0x1d, (byte) 0x7a, (byte) 0xcc, (byte) 0x16, (byte) 0x3c, (byte) 0xd1, (byte) 0x3b, (byte) 0x7e, (byte) 0x61, (byte) 0x53, (byte) 0xad, (byte) 0x5c, (byte) 0x45};
public static final byte[] up_right_turbo = new byte[]{(byte) 0xfb, (byte) 0x97, (byte) 0x6f, (byte) 0xba, (byte) 0x04, (byte) 0xaf, (byte) 0x87, (byte) 0x02, (byte) 0x22, (byte) 0x26, (byte) 0xec, (byte) 0x50, (byte) 0xae, (byte) 0x82, (byte) 0xf8, (byte) 0xc4};
public static final byte[] up_left_turbo = new byte[]{0x59, (byte) 0x23, (byte) 0x81, (byte) 0xc9, (byte) 0x43, (byte) 0xa4, (byte) 0x17, (byte) 0xca, (byte) 0x1b, (byte) 0xc3, (byte) 0xb5, (byte) 0x94, (byte) 0x00, (byte) 0xe0, (byte) 0xfc, (byte) 0x12};
public static final byte[] down_right_turbo = new byte[]{(byte) 0x80, (byte) 0xdf, (byte) 0xb2, (byte) 0x16, (byte) 0x5f, (byte) 0x32, (byte) 0x60, (byte) 0xf1, (byte) 0xd9, (byte) 0x83, (byte) 0x77, (byte) 0x50, (byte) 0xf4, (byte) 0x3a, (byte) 0x43, (byte) 0xda};
public static final byte[] down_left_turbo = new byte[]{(byte) 0xd5, (byte) 0x4a, (byte) 0xd5, (byte) 0x58, (byte) 0x57, (byte) 0xd3, (byte) 0x27, (byte) 0x74, (byte) 0x5f, (byte) 0x14, (byte) 0x1d, (byte) 0xd0, (byte) 0x0d, (byte) 0x67, (byte) 0x15, (byte) 0x95};
public static final byte[] idle_turbo_lights = new byte[]{(byte) 0x1a, (byte) 0x01, (byte) 0x1e, (byte) 0x9e, (byte) 0x6e, (byte) 0xfc, (byte) 0xce, (byte) 0x22, (byte) 0xbe, (byte) 0x8e, (byte) 0xb7, (byte) 0xff, (byte) 0xb6, (byte) 0x29, (byte) 0xfa, (byte) 0x75};
public static final byte[] up_turbo_lights = new byte[]{(byte) 0xd6, (byte) 0xc7, (byte) 0x63, (byte) 0x8e, (byte) 0x0e, (byte) 0x9a, (byte) 0x80, (byte) 0xbe, (byte) 0x60, (byte) 0x88, (byte) 0xfc, (byte) 0x44, (byte) 0x43, (byte) 0x07, (byte) 0xa1, (byte) 0x78};
public static final byte[] down_turbo_lights = new byte[]{(byte) 0x0d, (byte) 0x4f, (byte) 0xf8, (byte) 0x23, (byte) 0xac, (byte) 0xf9, (byte) 0xb7, (byte) 0xef, (byte) 0x1c, (byte) 0x26, (byte) 0xd4, (byte) 0xb4, (byte) 0x56, (byte) 0x51, (byte) 0x59, (byte) 0x52};
public static final byte[] up_right_turbo_lights = new byte[]{(byte) 0x83, (byte) 0xe6, (byte) 0x59, (byte) 0x42, (byte) 0x3d, (byte) 0x4b, (byte) 0x78, (byte) 0x48, (byte) 0x14, (byte) 0x5d, (byte) 0x86, (byte) 0xa1, (byte) 0x7b, (byte) 0x54, (byte) 0x7e, (byte) 0x58};
public static final byte[] up_left_turbo_lights = new byte[]{(byte) 0xd1, (byte) 0x7f, (byte) 0x6c, (byte) 0x5e, (byte) 0xe6, (byte) 0xba, (byte) 0x81, (byte) 0xe2, (byte) 0xb5, (byte) 0x80, (byte) 0x90, (byte) 0xa3, (byte) 0xcc, (byte) 0x76, (byte) 0x37, (byte) 0x9f};
public static final byte[] down_right_turbo_lights = new byte[]{(byte) 0xd4, (byte) 0x5d, (byte) 0xc9, (byte) 0x8f, (byte) 0x76, (byte) 0x58, (byte) 0xf9, (byte) 0x02, (byte) 0x0f, (byte) 0x93, (byte) 0xa0, (byte) 0xfd, (byte) 0x80, (byte) 0xe2, (byte) 0x2d, (byte) 0x45};
public static final byte[] down_left_turbo_lights = new byte[]{(byte) 0x96, (byte) 0xc0, (byte) 0x35, (byte) 0x77, (byte) 0xe9, (byte) 0xcd, (byte) 0xa8, (byte) 0xb9, (byte) 0x70, (byte) 0x21, (byte) 0x5f, (byte) 0xaf, (byte) 0x35, (byte) 0x00, (byte) 0x3b, (byte) 0x74};
public static final byte[] left_data = new byte[]{0x51, 0x38, 0x21, 0x12, 0x13, 0x5c, (byte) 0xcc, (byte) 0xdb, (byte) 0x46, (byte) 0xcf, (byte) 0x89, 0x21, (byte) 0xb7, 0x05, 0x49, (byte) 0x9a};
public static final byte[] right_data = new byte[]{0x1b, 0x57, 0x69, (byte) 0xcd, (byte) 0xf1, 0x3e, (byte) 0x8a, (byte) 0xb6, 0x27, 0x08, 0x0f, (byte) 0xf3, (byte) 0xce, (byte) 0xfc, 0x3b, (byte) 0xc0};
public enum SpeedModes {
NORMAL, TURBO, LIGHTS, TURBO_LIGHTS
} }
public enum Directions { public enum Light {
UP, DOWN, UP_LEFT, UP_RIGHT, DOWN_LEFT, DOWN_RIGHT ON, OFF
} }
public static byte[] get_directions_data(SpeedModes speedModes, Directions directions) { public enum Movement {
switch (speedModes) { UP, DOWN, IDLE
case NORMAL:
switch (directions) {
case UP:
return up_normal;
case UP_LEFT:
return up_left_normal;
case UP_RIGHT:
return up_right_normal;
case DOWN:
return down_normal;
case DOWN_LEFT:
return down_left_normal;
case DOWN_RIGHT:
return down_right_normal;
}
return idle_normal;
case TURBO:
switch (directions) {
case UP:
return up_turbo;
case UP_LEFT:
return up_left_turbo;
case UP_RIGHT:
return up_right_turbo;
case DOWN:
return down_turbo;
case DOWN_LEFT:
return down_left_turbo;
case DOWN_RIGHT:
return down_right_turbo;
}
return idle_turbo;
case LIGHTS:
switch (directions) {
case UP:
return up_lights;
case UP_LEFT:
return up_left_lights;
case UP_RIGHT:
return up_right_lights;
case DOWN:
return down_lights;
case DOWN_LEFT:
return down_left_lights;
case DOWN_RIGHT:
return down_right_lights;
}
return idle_lights;
case TURBO_LIGHTS:
switch (directions) {
case UP:
return up_turbo_lights;
case UP_LEFT:
return up_left_turbo_lights;
case UP_RIGHT:
return up_right_turbo_lights;
case DOWN:
return down_turbo_lights;
case DOWN_LEFT:
return down_left_turbo_lights;
case DOWN_RIGHT:
return down_right_turbo_lights;
}
return idle_turbo_lights;
}
return idle_lights;
} }
public enum Direction {
LEFT, RIGHT, CENTER
}
} }

View File

@ -37,7 +37,9 @@ public class SuperCarsSupport extends AbstractBTLEDeviceSupport {
private static final Logger LOG = LoggerFactory.getLogger(SuperCarsSupport.class); private static final Logger LOG = LoggerFactory.getLogger(SuperCarsSupport.class);
public static final String COMMAND_DRIVE_CONTROL = "nodomain.freeyourgadget.gadgetbridge.supercars.command.DRIVE_CONTROL"; public static final String COMMAND_DRIVE_CONTROL = "nodomain.freeyourgadget.gadgetbridge.supercars.command.DRIVE_CONTROL";
public static final String EXTRA_DIRECTION = "EXTRA_DIRECTION"; public static final String EXTRA_DIRECTION = "EXTRA_DIRECTION";
public static final String EXTRA_MODE = "EXTRA_MODE"; public static final String EXTRA_MOVEMENT = "EXTRA_MOVEMENT";
public static final String EXTRA_SPEED = "EXTRA_SPEED";
public static final String EXTRA_LIGHT = "EXTRA_LIGHT";
public SuperCarsSupport() { public SuperCarsSupport() {
super(LOG); super(LOG);
@ -48,7 +50,7 @@ public class SuperCarsSupport extends AbstractBTLEDeviceSupport {
protected TransactionBuilder initializeDevice(TransactionBuilder builder) { protected TransactionBuilder initializeDevice(TransactionBuilder builder) {
builder.add(new SetDeviceStateAction(getDevice(), GBDevice.State.INITIALIZING, getContext())); builder.add(new SetDeviceStateAction(getDevice(), GBDevice.State.INITIALIZING, getContext()));
builder.notify(getCharacteristic(SuperCarsConstants.CHARACTERISTIC_UUID_FFF4), true); //for battery builder.notify(getCharacteristic(SuperCarsConstants.CHARACTERISTIC_UUID_FFF4), true); //for battery
builder.setGattCallback(this);
LocalBroadcastManager broadcastManager = LocalBroadcastManager.getInstance(getContext()); LocalBroadcastManager broadcastManager = LocalBroadcastManager.getInstance(getContext());
IntentFilter filter = new IntentFilter(); IntentFilter filter = new IntentFilter();
@ -56,10 +58,8 @@ public class SuperCarsSupport extends AbstractBTLEDeviceSupport {
broadcastManager.registerReceiver(commandReceiver, filter); broadcastManager.registerReceiver(commandReceiver, filter);
getDevice().setFirmwareVersion("N/A"); getDevice().setFirmwareVersion("N/A");
getDevice().setFirmwareVersion2("N/A"); getDevice().setFirmwareVersion2("N/A");
builder.add(new SetDeviceStateAction(getDevice(), GBDevice.State.INITIALIZED, getContext())); builder.add(new SetDeviceStateAction(getDevice(), GBDevice.State.INITIALIZED, getContext()));
LOG.debug("name " + gbDevice.getName()); LOG.debug("Connected to: " + gbDevice.getName());
return builder; return builder;
} }
@ -67,10 +67,11 @@ public class SuperCarsSupport extends AbstractBTLEDeviceSupport {
@Override @Override
public void onReceive(Context context, Intent intent) { public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(COMMAND_DRIVE_CONTROL)) { if (intent.getAction().equals(COMMAND_DRIVE_CONTROL)) {
SuperCarsSupport.this.setDirection( send_data(
(SuperCarsConstants.SpeedModes) intent.getSerializableExtra(EXTRA_MODE), (SuperCarsConstants.Speed) intent.getSerializableExtra(EXTRA_SPEED),
(SuperCarsConstants.Directions) intent.getSerializableExtra(EXTRA_DIRECTION) (SuperCarsConstants.Movement) intent.getSerializableExtra(EXTRA_MOVEMENT),
(SuperCarsConstants.Light) intent.getSerializableExtra(EXTRA_LIGHT),
(SuperCarsConstants.Direction) intent.getSerializableExtra(EXTRA_DIRECTION)
); );
} }
} }
@ -79,24 +80,66 @@ public class SuperCarsSupport extends AbstractBTLEDeviceSupport {
@Override @Override
public boolean onCharacteristicChanged(BluetoothGatt gatt, public boolean onCharacteristicChanged(BluetoothGatt gatt,
BluetoothGattCharacteristic characteristic) { BluetoothGattCharacteristic characteristic) {
super.onCharacteristicChanged(gatt, characteristic); if (super.onCharacteristicChanged(gatt, characteristic)) {
byte[] data = characteristic.getValue(); return true;
byte[] decodedData = new byte[0];
try {
decodedData = CryptoUtils.decryptAES(data, SuperCarsConstants.aes_key);
} catch (Exception e) {
LOG.error("Error while decoding received data");
} }
if (decodedData.length == 16) { UUID characteristicUUID = characteristic.getUuid();
GBDeviceEventBatteryInfo batteryEvent = new GBDeviceEventBatteryInfo(); byte[] byte_data = characteristic.getValue();
batteryEvent.state = BatteryState.BATTERY_NORMAL; LOG.debug("got characteristics: " + characteristicUUID);
batteryEvent.level = decodedData[4]; /*
evaluateGBDeviceEvent(batteryEvent); //keep here for now. This requires descriptor reading
if (getDevice().getFirmwareVersion() == "N/A") {
// this is probably not the best way to retrieve UUID descriptor
// but i have not found a better way at this moment
BluetoothGattCharacteristic characteristic1 = getCharacteristic(SuperCarsConstants.CHARACTERISTIC_UUID_FFF1);
BluetoothGattDescriptor descriptor1 = characteristic1.getDescriptor(GattDescriptor.UUID_DESCRIPTOR_GATT_CHARACTERISTIC_USER_DESCRIPTION);
gatt.readDescriptor(descriptor1);
getDevice().setFirmwareVersion(new String(descriptor1.getValue()));
getDevice().sendDeviceUpdateIntent(getContext());
} else if (getDevice().getFirmwareVersion2() == "N/A") {
BluetoothGattCharacteristic characteristic2 = getCharacteristic(SuperCarsConstants.CHARACTERISTIC_UUID_FFF2);
BluetoothGattDescriptor descriptor2 = characteristic2.getDescriptor(GattDescriptor.UUID_DESCRIPTOR_GATT_CHARACTERISTIC_USER_DESCRIPTION);
gatt.readDescriptor(descriptor2);
//getDevice().setFirmwareVersion2(new String(descriptor2.getValue()));
//getDevice().sendDeviceUpdateIntent(getContext());
}
*/
if (SuperCarsConstants.CHARACTERISTIC_UUID_FFF4.equals(characteristicUUID)) {
byte[] decoded_data = decryptData(byte_data);
if (decoded_data.length == 16) {
GBDeviceEventBatteryInfo batteryEvent = new GBDeviceEventBatteryInfo();
batteryEvent.state = BatteryState.BATTERY_NORMAL;
batteryEvent.level = decoded_data[4];
evaluateGBDeviceEvent(batteryEvent);
}
} }
return true; return true;
} }
private byte[] decryptData(byte[] data) {
byte[] decodedResult = new byte[0];
try {
decodedResult = CryptoUtils.decryptAES(data, SuperCarsConstants.aes_key);
} catch (Exception e) {
LOG.error("Error while decoding received data: " + e);
}
return decodedResult;
}
private byte[] encryptData(byte[] data) {
byte[] encodedResult = new byte[0];
try {
encodedResult = CryptoUtils.encryptAES(data, SuperCarsConstants.aes_key);
} catch (Exception e) {
LOG.error("Error while encoding data to be sent: " + e);
}
return encodedResult;
}
@Override @Override
public void onNotification(NotificationSpec notificationSpec) { public void onNotification(NotificationSpec notificationSpec) {
@ -251,15 +294,60 @@ public class SuperCarsSupport extends AbstractBTLEDeviceSupport {
return false; return false;
} }
private void setDirection(SuperCarsConstants.SpeedModes speedModes, SuperCarsConstants.Directions direction) { private void send_data(SuperCarsConstants.Speed speed,
SuperCarsConstants.Movement movement,
SuperCarsConstants.Light light,
SuperCarsConstants.Direction direction) {
byte[] command = SuperCarsConstants.get_directions_data(speedModes, direction); byte[] command = craft_packet(speed, direction, movement, light);
TransactionBuilder builder = new TransactionBuilder("setDirections"); TransactionBuilder builder = new TransactionBuilder("send data");
BluetoothGattCharacteristic writeCharacteristic = getCharacteristic(SuperCarsConstants.CHARACTERISTIC_UUID_FFF1); BluetoothGattCharacteristic writeCharacteristic = getCharacteristic(SuperCarsConstants.CHARACTERISTIC_UUID_FFF1);
builder.write(writeCharacteristic, command); builder.write(writeCharacteristic, encryptData(command));
builder.queue(getQueue()); builder.queue(getQueue());
} }
private byte[] craft_packet(SuperCarsConstants.Speed speed,
SuperCarsConstants.Direction direction,
SuperCarsConstants.Movement movement,
SuperCarsConstants.Light light) {
byte[] packet = new byte[]{0x0, 0x43, 0x54, 0x4c, 0x0, 0x0, 0x0, 0x0, 0x1, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0};
switch (movement) {
case UP:
packet[4] = 0x1;
break;
case DOWN:
packet[5] = 0x1;
break;
}
switch (direction) {
case LEFT:
packet[6] = 0x1;
break;
case RIGHT:
packet[7] = 0x1;
break;
}
switch (light) {
case ON:
packet[8] = 0x0;
break;
case OFF:
packet[8] = 0x1;
break;
}
switch (speed) {
case NORMAL:
packet[9] = 0x50;
break;
case TURBO:
packet[9] = 0x64;
break;
}
return packet;
}
@Override @Override
public void dispose() { public void dispose() {
super.dispose(); super.dispose();

View File

@ -26,14 +26,14 @@
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_weight="1" android:layout_weight="1"
android:text="TurboMode" /> android:text="@string/supercars_turbo_speed_label" />
<CheckBox <CheckBox
android:id="@+id/lightsOn" android:id="@+id/lightsOn"
android:layout_width="match_parent" android:layout_width="match_parent"
android:layout_height="wrap_content" android:layout_height="wrap_content"
android:layout_weight="1" android:layout_weight="1"
android:text="Lights" /> android:text="@string/supercars_lights_label" />
</LinearLayout> </LinearLayout>
</LinearLayout> </LinearLayout>

View File

@ -1845,7 +1845,9 @@
<string name="prefs_hourly_chime">Hourly chime</string> <string name="prefs_hourly_chime">Hourly chime</string>
<string name="prefs_hourly_chime_summary">The watch will beep once an hour</string> <string name="prefs_hourly_chime_summary">The watch will beep once an hour</string>
<string name="devicetype_flipper_zero">Flipper zero</string> <string name="devicetype_flipper_zero">Flipper zero</string>
<string name="devicetype_super_cars">Shell Racing</string>
<string name="activity_prefs_allow_bluetooth_intent_api">Bluetooth Intent API</string> <string name="activity_prefs_allow_bluetooth_intent_api">Bluetooth Intent API</string>
<string name="activity_prefs_summary_allow_bluetooth_intent_api">Allow controlling Bluetooth connection via Intent API</string> <string name="activity_prefs_summary_allow_bluetooth_intent_api">Allow controlling Bluetooth connection via Intent API</string>
<string name="devicetype_super_cars">Shell Racing</string>
<string name="supercars_turbo_speed_label">Turbo Speed</string>
<string name="supercars_lights_label">Lights</string>
</resources> </resources>