184 lines
6.6 KiB
Python
184 lines
6.6 KiB
Python
import asyncio
|
|
import json
|
|
import random
|
|
from ulab import numpy as np
|
|
|
|
import arena
|
|
import robot
|
|
|
|
class DistanceSensorTracker:
|
|
def __init__(self):
|
|
robot.left_distance.distance_mode = 2
|
|
robot.right_distance.distance_mode = 2
|
|
robot.left_distance.timing_budget = 50
|
|
robot.right_distance.timing_budget = 50
|
|
self.left = 300
|
|
self.right = 300
|
|
|
|
async def main(self):
|
|
robot.left_distance.start_ranging()
|
|
robot.right_distance.start_ranging()
|
|
while True:
|
|
if robot.left_distance.data_ready and robot.left_distance.distance:
|
|
self.left = robot.left_distance.distance * 10 # convert to mm
|
|
robot.left_distance.clear_interrupt()
|
|
if robot.right_distance.data_ready and robot.right_distance.distance:
|
|
self.right = robot.right_distance.distance * 10
|
|
robot.right_distance.clear_interrupt()
|
|
await asyncio.sleep(0.01)
|
|
|
|
|
|
class CollisionAvoid:
|
|
def __init__(self, distance_sensors):
|
|
self.speed = 0.6
|
|
self.distance_sensors = distance_sensors
|
|
|
|
async def main(self):
|
|
while True:
|
|
robot.set_right(self.speed)
|
|
while self.distance_sensors.left < 300 or \
|
|
self.distance_sensors.right < 300:
|
|
robot.set_left(-self.speed)
|
|
await asyncio.sleep(0.3)
|
|
robot.set_left(self.speed)
|
|
await asyncio.sleep(0)
|
|
|
|
def get_scaled_sample_around_mean(mean, scale):
|
|
return mean + (random.uniform(-scale, scale) + random.uniform(-scale, scale)) / 2
|
|
|
|
def send_json(data):
|
|
robot.uart.write((json.dumps(data) + "\n").encode())
|
|
|
|
def read_json():
|
|
try:
|
|
data = robot.uart.readline()
|
|
decoded = data.decode()
|
|
return json.loads(decoded)
|
|
except (UnicodeError, ValueError):
|
|
print("Invalid data")
|
|
return None
|
|
|
|
|
|
def send_poses(samples):
|
|
send_json({
|
|
"poses": np.array(samples[:,:2], dtype=np.int16).tolist(),
|
|
})
|
|
|
|
|
|
class Simulation:
|
|
def __init__(self):
|
|
self.population_size = 20
|
|
self.poses = np.array(
|
|
[(
|
|
int(random.uniform(0, arena.width)),
|
|
int(random.uniform(0, arena.height)),
|
|
int(random.uniform(0, 360))) for _ in range(self.population_size)],
|
|
dtype=np.float,
|
|
)
|
|
self.distance_sensors = DistanceSensorTracker()
|
|
self.collision_avoider = CollisionAvoid(self.distance_sensors)
|
|
self.last_encoder_left = robot.left_encoder.read()
|
|
self.last_encoder_right = robot.right_encoder.read()
|
|
self.alpha_rot = 0.05
|
|
self.alpha_rot_trans = 0.01
|
|
self.alpha_trans = 0.05
|
|
self.alpha_trans_rot = 0.01
|
|
|
|
def convert_odometry_to_motion(self, left_encoder_delta, right_encoder_delta):
|
|
"""
|
|
left_encoder is the change in the left encoder
|
|
right_encoder is the change in the right encoder
|
|
returns rot1, trans, rot2
|
|
rot1 is the rotation of the robot in degrees before the translation
|
|
trans is the distance the robot has moved in mm
|
|
rot2 is the rotation of the robot in degrees
|
|
"""
|
|
left_mm = left_encoder_delta * robot.ticks_to_mm
|
|
right_mm = right_encoder_delta * robot.ticks_to_mm
|
|
|
|
if left_mm == right_mm:
|
|
return 0, left_mm, 0
|
|
|
|
# calculate the radius of the arc
|
|
radius = (robot.wheelbase_mm / 2) * (left_mm + right_mm) / (right_mm - left_mm)
|
|
## angle = difference in steps / wheelbase
|
|
d_theta = (right_mm - left_mm) / robot.wheelbase_mm
|
|
# For a small enough motion, assume that the chord length = arc length
|
|
arc_length = d_theta * radius
|
|
rot1 = np.degrees(d_theta/2)
|
|
rot2 = rot1
|
|
return rot1, arc_length, rot2
|
|
|
|
def apply_motion_to_poses(self, rot1, trans, rot2):
|
|
self.poses[:,2] += rot1
|
|
rot1_radians = np.radians(self.poses[:,2])
|
|
self.poses[:,0] += trans * np.cos(rot1_radians)
|
|
self.poses[:,1] += trans * np.sin(rot1_radians)
|
|
self.poses[:,2] += rot2
|
|
self.poses[:,2] = np.array([float(theta % 360) for theta in self.poses[:,2]])
|
|
|
|
def apply_randomness_to_motion(self, rot1, trans, rot2):
|
|
rot1_scale = self.alpha_rot * abs(rot1) + self.alpha_rot_trans * abs(trans)
|
|
trans_scale = self.alpha_trans * abs(trans) + self.alpha_trans_rot * (abs(rot1) + abs(rot2))
|
|
rot2_scale = self.alpha_rot * abs(rot2) + self.alpha_rot_trans * abs(trans)
|
|
rot1_model = np.array([get_scaled_sample_around_mean(rot1, rot1_scale) for _ in range(self.poses.shape[0])])
|
|
trans_model = np.array([get_scaled_sample_around_mean(trans, trans_scale) for _ in range(self.poses.shape[0])])
|
|
rot2_model = np.array([get_scaled_sample_around_mean(rot2, rot2_scale) for _ in range(self.poses.shape[0])])
|
|
return rot1_model, trans_model, rot2_model
|
|
|
|
def motion_model(self):
|
|
"""Apply the motion model"""
|
|
new_encoder_left = robot.left_encoder.read()
|
|
new_encoder_right = robot.right_encoder.read()
|
|
|
|
rot1, trans, rot2 = self.convert_odometry_to_motion(
|
|
new_encoder_left - self.last_encoder_left,
|
|
new_encoder_right - self.last_encoder_right)
|
|
self.last_encoder_left = new_encoder_left
|
|
self.last_encoder_right = new_encoder_right
|
|
|
|
rot1_model, trans_model, rot2_model = self.apply_randomness_to_motion(rot1, trans, rot2)
|
|
self.apply_motion_to_poses(rot1_model, trans_model, rot2_model)
|
|
|
|
print(
|
|
json.dumps(
|
|
[self.poses.tolist(), rot1, trans, rot2]
|
|
)
|
|
)
|
|
|
|
async def main(self):
|
|
asyncio.create_task(self.distance_sensors.main())
|
|
collision_avoider = asyncio.create_task(self.collision_avoider.main())
|
|
try:
|
|
while True:
|
|
send_poses(self.poses)
|
|
await asyncio.sleep(0.05)
|
|
self.motion_model()
|
|
finally:
|
|
collision_avoider.cancel()
|
|
robot.stop()
|
|
|
|
|
|
async def command_handler(simulation):
|
|
print("Starting handler")
|
|
simulation_task = None
|
|
while True:
|
|
if robot.uart.in_waiting:
|
|
request = read_json()
|
|
if not request:
|
|
continue
|
|
print("Received: ", request)
|
|
if request["command"] == "arena":
|
|
send_json({
|
|
"arena": arena.boundary_lines,
|
|
})
|
|
elif request["command"] == "start":
|
|
if not simulation_task:
|
|
simulation_task = asyncio.create_task(simulation.main())
|
|
|
|
await asyncio.sleep(0.1)
|
|
|
|
|
|
simulation = Simulation()
|
|
asyncio.run(command_handler(simulation))
|