You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
163 lines
5.7 KiB
163 lines
5.7 KiB
2 years ago
|
import numpy as np
|
||
|
|
||
|
from openpilot.common.params import Params
|
||
|
from openpilot.tools.sim.lib.common import SimulatorState, vec3
|
||
|
from openpilot.tools.sim.bridge.common import World, SimulatorBridge
|
||
|
from openpilot.tools.sim.lib.camerad import W, H
|
||
|
|
||
|
|
||
|
class CarlaWorld(World):
|
||
|
def __init__(self, world, vehicle, high_quality=False, dual_camera=False):
|
||
|
super().__init__(dual_camera)
|
||
|
import carla
|
||
|
self.world = world
|
||
|
self.vc: carla.VehicleControl = carla.VehicleControl(throttle=0, steer=0, brake=0, reverse=False)
|
||
|
self.vehicle = vehicle
|
||
|
self.max_steer_angle: float = vehicle.get_physics_control().wheels[0].max_steer_angle
|
||
|
self.params = Params()
|
||
|
|
||
|
self.steer_ratio = 15
|
||
|
|
||
|
self.carla_objects = []
|
||
|
|
||
|
blueprint_library = self.world.get_blueprint_library()
|
||
|
transform = carla.Transform(carla.Location(x=0.8, z=1.13))
|
||
|
|
||
|
def create_camera(fov, callback):
|
||
|
blueprint = blueprint_library.find('sensor.camera.rgb')
|
||
|
blueprint.set_attribute('image_size_x', str(W))
|
||
|
blueprint.set_attribute('image_size_y', str(H))
|
||
|
blueprint.set_attribute('fov', str(fov))
|
||
|
blueprint.set_attribute('sensor_tick', str(1/20))
|
||
|
if not high_quality:
|
||
|
blueprint.set_attribute('enable_postprocess_effects', 'False')
|
||
|
camera = world.spawn_actor(blueprint, transform, attach_to=vehicle)
|
||
|
camera.listen(callback)
|
||
|
return camera
|
||
|
|
||
|
self.road_camera = create_camera(fov=40, callback=self.cam_callback_road)
|
||
|
if dual_camera:
|
||
|
self.road_wide_camera = create_camera(fov=120, callback=self.cam_callback_wide_road) # fov bigger than 120 shows unwanted artifacts
|
||
|
else:
|
||
|
self.road_wide_camera = None
|
||
|
|
||
|
# re-enable IMU
|
||
|
imu_bp = blueprint_library.find('sensor.other.imu')
|
||
|
imu_bp.set_attribute('sensor_tick', '0.01')
|
||
|
self.imu = world.spawn_actor(imu_bp, transform, attach_to=vehicle)
|
||
|
|
||
|
gps_bp = blueprint_library.find('sensor.other.gnss')
|
||
|
self.gps = world.spawn_actor(gps_bp, transform, attach_to=vehicle)
|
||
|
self.params.put_bool("UbloxAvailable", True)
|
||
|
|
||
|
self.carla_objects = [self.imu, self.gps, self.road_camera, self.road_wide_camera]
|
||
|
|
||
|
def close(self):
|
||
|
for s in self.carla_objects:
|
||
|
if s is not None:
|
||
|
try:
|
||
|
s.destroy()
|
||
|
except Exception as e:
|
||
|
print("Failed to destroy carla object", e)
|
||
|
|
||
|
def carla_image_to_rgb(self, image):
|
||
|
rgb = np.frombuffer(image.raw_data, dtype=np.dtype("uint8"))
|
||
|
rgb = np.reshape(rgb, (H, W, 4))
|
||
|
return np.ascontiguousarray(rgb[:, :, [0, 1, 2]])
|
||
|
|
||
|
def cam_callback_road(self, image):
|
||
|
with self.image_lock:
|
||
|
self.road_image = self.carla_image_to_rgb(image)
|
||
|
|
||
|
def cam_callback_wide_road(self, image):
|
||
|
with self.image_lock:
|
||
|
self.wide_road_image = self.carla_image_to_rgb(image)
|
||
|
|
||
|
def apply_controls(self, steer_angle, throttle_out, brake_out):
|
||
|
self.vc.throttle = throttle_out
|
||
|
|
||
|
steer_carla = steer_angle * -1 / (self.max_steer_angle * self.steer_ratio)
|
||
|
steer_carla = np.clip(steer_carla, -1, 1)
|
||
|
|
||
|
self.vc.steer = steer_carla
|
||
|
self.vc.brake = brake_out
|
||
|
self.vehicle.apply_control(self.vc)
|
||
|
|
||
|
def read_sensors(self, simulator_state: SimulatorState):
|
||
|
simulator_state.imu.bearing = self.imu.get_transform().rotation.yaw
|
||
|
|
||
|
simulator_state.imu.accelerometer = vec3(
|
||
|
self.imu.get_acceleration().x,
|
||
|
self.imu.get_acceleration().y,
|
||
|
self.imu.get_acceleration().z
|
||
|
)
|
||
|
|
||
|
simulator_state.imu.gyroscope = vec3(
|
||
|
self.imu.get_angular_velocity().x,
|
||
|
self.imu.get_angular_velocity().y,
|
||
|
self.imu.get_angular_velocity().z
|
||
|
)
|
||
|
|
||
|
simulator_state.gps.from_xy([self.vehicle.get_location().x, self.vehicle.get_location().y])
|
||
|
|
||
|
simulator_state.velocity = self.vehicle.get_velocity()
|
||
|
simulator_state.valid = True
|
||
|
simulator_state.steering_angle = self.vc.steer * self.max_steer_angle
|
||
|
|
||
|
def read_cameras(self):
|
||
|
pass # cameras are read within a callback for carla
|
||
|
|
||
|
def tick(self):
|
||
|
self.world.tick()
|
||
|
|
||
|
|
||
|
class CarlaBridge(SimulatorBridge):
|
||
|
TICKS_PER_FRAME = 5
|
||
|
|
||
|
def __init__(self, arguments):
|
||
|
super().__init__(arguments)
|
||
|
self.host = arguments.host
|
||
|
self.port = arguments.port
|
||
|
self.town = arguments.town
|
||
|
self.num_selected_spawn_point = arguments.num_selected_spawn_point
|
||
|
|
||
|
def spawn_world(self):
|
||
|
import carla
|
||
|
client = carla.Client(self.host, self.port)
|
||
|
client.set_timeout(5)
|
||
|
|
||
|
world = client.load_world(self.town)
|
||
|
|
||
|
settings = world.get_settings()
|
||
|
settings.fixed_delta_seconds = 0.01
|
||
|
world.apply_settings(settings)
|
||
|
|
||
|
world.set_weather(carla.WeatherParameters.ClearSunset)
|
||
|
|
||
|
if not self.high_quality:
|
||
|
world.unload_map_layer(carla.MapLayer.Foliage)
|
||
|
world.unload_map_layer(carla.MapLayer.Buildings)
|
||
|
world.unload_map_layer(carla.MapLayer.ParkedVehicles)
|
||
|
world.unload_map_layer(carla.MapLayer.Props)
|
||
|
world.unload_map_layer(carla.MapLayer.StreetLights)
|
||
|
world.unload_map_layer(carla.MapLayer.Particles)
|
||
|
|
||
|
blueprint_library = world.get_blueprint_library()
|
||
|
|
||
|
world_map = world.get_map()
|
||
|
|
||
|
vehicle_bp = blueprint_library.filter('vehicle.tesla.*')[1]
|
||
|
vehicle_bp.set_attribute('role_name', 'hero')
|
||
|
spawn_points = world_map.get_spawn_points()
|
||
|
assert len(spawn_points) > self.num_selected_spawn_point, \
|
||
|
f'''No spawn point {self.num_selected_spawn_point}, try a value between 0 and {len(spawn_points)} for this town.'''
|
||
|
spawn_point = spawn_points[self.num_selected_spawn_point]
|
||
|
vehicle = world.spawn_actor(vehicle_bp, spawn_point)
|
||
|
|
||
|
physics_control = vehicle.get_physics_control()
|
||
|
physics_control.mass = 2326
|
||
|
physics_control.torque_curve = [[20.0, 500.0], [5000.0, 500.0]]
|
||
|
physics_control.gear_switch_time = 0.0
|
||
|
vehicle.apply_physics_control(physics_control)
|
||
|
|
||
|
return CarlaWorld(world, vehicle, dual_camera=self.dual_camera)
|