Source code for scenic.simulators.newtonian.simulator

"""Newtonian simulator implementation."""

from cmath import atan, pi, tan
import math
from math import copysign, degrees, radians, sin
import os
import pathlib
import time

import scenic.core.errors as errors  # isort: skip

if errors.verbosityLevel == 0:  # suppress pygame advertisement at zero verbosity
    os.environ["PYGAME_HIDE_SUPPORT_PROMPT"] = "hide"
import pygame
import shapely

from scenic.core.geometry import allChains, findMinMax
from scenic.core.regions import toPolygon
from scenic.core.simulators import SimulationCreationError
from scenic.core.vectors import Orientation, Vector
from scenic.domains.driving.controllers import (
    PIDLateralController,
    PIDLongitudinalController,
)
from scenic.domains.driving.roads import Network
from scenic.domains.driving.simulators import DrivingSimulation, DrivingSimulator
from scenic.syntax.veneer import verbosePrint

current_dir = pathlib.Path(__file__).parent.absolute()

WIDTH = 1280
HEIGHT = 800
MAX_ACCELERATION = 5.6  # in m/s2, seems to be a pretty reasonable value
MAX_BRAKING = 4.6

ROAD_COLOR = (0, 0, 0)
ROAD_WIDTH = 2
LANE_COLOR = (96, 96, 96)
CENTERLINE_COLOR = (224, 224, 224)
SIDEWALK_COLOR = (0, 128, 255)
SHOULDER_COLOR = (96, 96, 96)


[docs]class NewtonianSimulator(DrivingSimulator): """Implementation of `Simulator` for the Newtonian simulator. Args: network (Network): road network to display in the background, if any. render (bool): whether to render the simulation in a window. .. versionchanged:: 3.0 The **timestep** argument is removed: it can be specified when calling `simulate` instead. The default timestep for the Newtonian simulator when not otherwise specified is still 0.1 seconds. """ def __init__(self, network=None, render=False): super().__init__() self.render = render self.network = network def createSimulation(self, scene, **kwargs): return NewtonianSimulation(scene, self.network, self.render, **kwargs)
[docs]class NewtonianSimulation(DrivingSimulation): """Implementation of `Simulation` for the Newtonian simulator.""" def __init__(self, scene, network, render, timestep, **kwargs): self.render = render self.network = network if timestep is None: timestep = 0.1 super().__init__(scene, timestep=timestep, **kwargs) def setup(self): super().setup() if self.render: # determine window size min_x, max_x = findMinMax(obj.x for obj in self.objects) min_y, max_y = findMinMax(obj.y for obj in self.objects) pygame.init() pygame.font.init() self.screen = pygame.display.set_mode( (WIDTH, HEIGHT), pygame.HWSURFACE | pygame.DOUBLEBUF ) self.screen.fill((255, 255, 255)) x, y, _ = self.objects[0].position self.min_x, self.max_x = min_x - 50, max_x + 50 self.min_y, self.max_y = min_y - 50, max_y + 50 self.size_x = self.max_x - self.min_x self.size_y = self.max_y - self.min_y self.screen_poly = shapely.geometry.Polygon( ( (self.min_x, self.min_y), (self.max_x, self.min_y), (self.max_x, self.max_y), (self.min_x, self.max_y), ) ) img_path = os.path.join(current_dir, "car.png") self.car = pygame.image.load(img_path) self.car_width = int(3.5 * WIDTH / self.size_x) self.car_height = self.car_width self.car = pygame.transform.scale(self.car, (self.car_width, self.car_height)) self.parse_network() self.draw_objects() def parse_network(self): self.network_polygons = [] if not self.network: return def addRegion(region, color, width=1): poly = toPolygon(region) if not poly or not self.screen_poly.intersects(poly): return for chain in allChains(poly): coords = tuple(self.scenicToScreenVal(pt) for pt in chain.coords) self.network_polygons.append((coords, color, width)) addRegion(self.network.walkableRegion, SIDEWALK_COLOR) addRegion(self.network.shoulderRegion, SHOULDER_COLOR) for road in self.network.roads: # loop over main roads for lane in road.lanes: addRegion(lane.leftEdge, LANE_COLOR) addRegion(lane.rightEdge, LANE_COLOR) addRegion(road, ROAD_COLOR, ROAD_WIDTH) for lane in self.network.lanes: # loop over all lanes, even in intersections addRegion(lane.centerline, CENTERLINE_COLOR) addRegion(self.network.intersectionRegion, ROAD_COLOR) def scenicToScreenVal(self, pos): x, y = pos[:2] x_prop = (x - self.min_x) / self.size_x y_prop = (y - self.min_y) / self.size_y return int(x_prop * WIDTH), HEIGHT - 1 - int(y_prop * HEIGHT) def createObjectInSimulator(self, obj): # Set actor's initial speed obj.speed = math.hypot(*obj.velocity) if hasattr(obj, "elevation"): obj.elevation = 0.0 def isOnScreen(self, x, y): return self.min_x <= x <= self.max_x and self.min_y <= y <= self.max_y def step(self): for obj in self.objects: current_speed = obj.velocity.norm() if hasattr(obj, "hand_brake"): forward = obj.velocity.dot(Vector(0, 1).rotatedBy(obj.heading)) >= 0 signed_speed = current_speed if forward else -current_speed if obj.hand_brake or obj.brake > 0: braking = MAX_BRAKING * max(obj.hand_brake, obj.brake) acceleration = braking * self.timestep if acceleration >= current_speed: signed_speed = 0 elif forward: signed_speed -= acceleration else: signed_speed += acceleration else: acceleration = obj.throttle * MAX_ACCELERATION if obj.reverse: acceleration *= -1 signed_speed += acceleration * self.timestep obj.velocity = Vector(0, signed_speed).rotatedBy(obj.heading) if obj.steer: turning_radius = obj.length / sin(obj.steer * math.pi / 2) obj.angularSpeed = -signed_speed / turning_radius else: obj.angularSpeed = 0 obj.speed = abs(signed_speed) else: obj.speed = current_speed obj.position += obj.velocity * self.timestep obj.heading += obj.angularSpeed * self.timestep if self.render: self.draw_objects() pygame.event.pump() def draw_objects(self): self.screen.fill((255, 255, 255)) for screenPoints, color, width in self.network_polygons: pygame.draw.lines(self.screen, color, False, screenPoints, width=width) for i, obj in enumerate(self.objects): color = (255, 0, 0) if i == 0 else (0, 0, 255) h, w = obj.length, obj.width pos_vec = Vector(-1.75, 1.75) neg_vec = Vector(w / 2, h / 2) heading_vec = Vector(0, 10).rotatedBy(obj.heading) dx, dy = int(heading_vec.x), -int(heading_vec.y) x, y = self.scenicToScreenVal(obj.position) rect_x, rect_y = self.scenicToScreenVal(obj.position + pos_vec) if hasattr(obj, "isCar") and obj.isCar: self.rotated_car = pygame.transform.rotate( self.car, math.degrees(obj.heading) ) self.screen.blit(self.rotated_car, (rect_x, rect_y)) else: corners = [self.scenicToScreenVal(corner) for corner in obj._corners2D] pygame.draw.polygon(self.screen, color, corners) pygame.display.update() time.sleep(self.timestep) def getProperties(self, obj, properties): yaw, _, _ = obj.parentOrientation.globalToLocalAngles(obj.heading, 0, 0) values = dict( position=obj.position, yaw=yaw, pitch=0, roll=0, velocity=obj.velocity, speed=obj.speed, angularSpeed=obj.angularSpeed, angularVelocity=obj.angularVelocity, ) if "elevation" in properties: values["elevation"] = obj.elevation return values def destroy(self): if self.render: pygame.quit() def getLaneFollowingControllers(self, agent): dt = self.timestep if agent.isCar: lon_controller = PIDLongitudinalController(K_P=0.5, K_D=0.1, K_I=0.7, dt=dt) lat_controller = PIDLateralController(K_P=0.1, K_D=0.1, K_I=0.02, dt=dt) else: lon_controller = PIDLongitudinalController( K_P=0.25, K_D=0.025, K_I=0.0, dt=dt ) lat_controller = PIDLateralController(K_P=0.2, K_D=0.1, K_I=0.0, dt=dt) return lon_controller, lat_controller def getTurningControllers(self, agent): dt = self.timestep if agent.isCar: lon_controller = PIDLongitudinalController(K_P=0.5, K_D=0.1, K_I=0.7, dt=dt) lat_controller = PIDLateralController(K_P=0.2, K_D=0.2, K_I=0.2, dt=dt) else: lon_controller = PIDLongitudinalController( K_P=0.25, K_D=0.025, K_I=0.0, dt=dt ) lat_controller = PIDLateralController(K_P=0.4, K_D=0.1, K_I=0.0, dt=dt) return lon_controller, lat_controller def getLaneChangingControllers(self, agent): dt = self.timestep if agent.isCar: lon_controller = PIDLongitudinalController(K_P=0.5, K_D=0.1, K_I=0.7, dt=dt) lat_controller = PIDLateralController(K_P=0.2, K_D=0.2, K_I=0.02, dt=dt) else: lon_controller = PIDLongitudinalController( K_P=0.25, K_D=0.025, K_I=0.0, dt=dt ) lat_controller = PIDLateralController(K_P=0.1, K_D=0.3, K_I=0.0, dt=dt) return lon_controller, lat_controller