osrm-backend/profiles/car.lua

438 lines
11 KiB
Lua
Raw Normal View History

-- Car profile
2017-05-18 08:27:28 -04:00
api_version = 2
Set = require('lib/set')
Sequence = require('lib/sequence')
Handlers = require("lib/way_handlers")
find_access_tag = require("lib/access").find_access_tag
limit = require("lib/maxspeed").limit
function setup()
local use_left_hand_driving = false
return {
properties = {
max_speed_for_map_matching = 180/3.6, -- 180kmph -> m/s
left_hand_driving = use_left_hand_driving,
-- For routing based on duration, but weighted for preferring certain roads
weight_name = 'routability',
-- For shortest duration without penalties for accessibility
-- weight_name = 'duration',
-- For shortest distance without penalties for accessibility
-- weight_name = 'distance',
process_call_tagless_node = false,
u_turn_penalty = 20,
continue_straight_at_waypoint = true,
use_turn_restrictions = true,
traffic_light_penalty = 2,
},
2017-07-19 09:32:42 -04:00
2017-05-18 08:27:28 -04:00
default_mode = mode.driving,
default_speed = 10,
oneway_handling = true,
side_road_multiplier = 0.8,
turn_penalty = 7.5,
speed_reduction = 0.8,
-- Note: this biases right-side driving.
-- Should be inverted for left-driving countries.
turn_bias = use_left_hand_driving and 1/1.075 or 1.075,
-- a list of suffixes to suppress in name change instructions
suffix_list = {
'N', 'NE', 'E', 'SE', 'S', 'SW', 'W', 'NW', 'North', 'South', 'West', 'East'
},
barrier_whitelist = Set {
'cattle_grid',
'border_control',
'toll_booth',
'sally_port',
'gate',
'lift_gate',
'no',
'entrance'
},
access_tag_whitelist = Set {
'yes',
'motorcar',
'motor_vehicle',
'vehicle',
'permissive',
'designated',
'hov'
},
access_tag_blacklist = Set {
'no',
'agricultural',
'forestry',
'emergency',
'psv',
'customers',
'private',
'delivery',
'destination'
},
restricted_access_tag_list = Set {
'private',
'delivery',
'destination',
'customers',
},
access_tags_hierarchy = Sequence {
'motorcar',
'motor_vehicle',
'vehicle',
'access'
},
service_tag_forbidden = Set {
'emergency_access'
},
restrictions = Sequence {
'motorcar',
'motor_vehicle',
'vehicle'
},
2017-08-14 18:18:57 -04:00
classes = Sequence {
'toll', 'motorway', 'ferry', 'restricted'
},
2017-08-16 16:21:19 -04:00
-- classes to support for exclude flags
excludable = Sequence {
2017-08-14 18:18:57 -04:00
Set {'toll'},
Set {'motorway'},
Set {'ferry'}
},
2017-05-18 08:27:28 -04:00
avoid = Set {
'area',
-- 'toll', -- uncomment this to avoid tolls
'reversible',
'impassable',
'hov_lanes',
'steps',
'construction',
'proposed'
},
speeds = Sequence {
highway = {
motorway = 90,
motorway_link = 45,
trunk = 85,
trunk_link = 40,
primary = 65,
primary_link = 30,
secondary = 55,
secondary_link = 25,
tertiary = 40,
tertiary_link = 20,
unclassified = 25,
residential = 25,
living_street = 10,
service = 15
}
},
service_penalties = {
alley = 0.5,
parking = 0.5,
parking_aisle = 0.5,
driveway = 0.5,
["drive-through"] = 0.5,
["drive-thru"] = 0.5
},
2017-07-19 09:32:42 -04:00
restricted_highway_whitelist = Set {
'motorway',
'motorway_link',
'trunk',
'trunk_link',
'primary',
'primary_link',
'secondary',
'secondary_link',
'tertiary',
'tertiary_link',
'residential',
'living_street',
2017-05-18 08:27:28 -04:00
},
construction_whitelist = Set {
'no',
'widening',
'minor',
},
2017-05-18 08:27:28 -04:00
route_speeds = {
ferry = 5,
shuttle_train = 10
},
bridge_speeds = {
movable = 5
},
-- surface/trackype/smoothness
-- values were estimated from looking at the photos at the relevant wiki pages
-- max speed for surfaces
surface_speeds = {
asphalt = nil, -- nil mean no limit. removing the line has the same effect
concrete = nil,
["concrete:plates"] = nil,
["concrete:lanes"] = nil,
paved = nil,
cement = 80,
compacted = 80,
fine_gravel = 80,
paving_stones = 60,
metal = 60,
bricks = 60,
grass = 40,
wood = 40,
sett = 40,
grass_paver = 40,
gravel = 40,
unpaved = 40,
ground = 40,
dirt = 40,
pebblestone = 40,
tartan = 40,
cobblestone = 30,
clay = 30,
earth = 20,
stone = 20,
rocky = 20,
sand = 20,
mud = 10
},
-- max speed for tracktypes
tracktype_speeds = {
grade1 = 60,
grade2 = 40,
grade3 = 30,
grade4 = 25,
grade5 = 20
},
-- max speed for smoothnesses
smoothness_speeds = {
intermediate = 80,
bad = 40,
very_bad = 20,
horrible = 10,
very_horrible = 5,
impassable = 0
},
-- http://wiki.openstreetmap.org/wiki/Speed_limits
maxspeed_table_default = {
urban = 50,
rural = 90,
trunk = 110,
motorway = 130
},
-- List only exceptions
maxspeed_table = {
["ch:rural"] = 80,
["ch:trunk"] = 100,
["ch:motorway"] = 120,
["de:living_street"] = 7,
["ru:living_street"] = 20,
["ru:urban"] = 60,
["ua:urban"] = 60,
["at:rural"] = 100,
["de:rural"] = 100,
["at:trunk"] = 100,
["cz:trunk"] = 0,
["ro:trunk"] = 100,
["cz:motorway"] = 0,
["de:motorway"] = 0,
["ru:motorway"] = 110,
["gb:nsl_single"] = (60*1609)/1000,
["gb:nsl_dual"] = (70*1609)/1000,
["gb:motorway"] = (70*1609)/1000,
["uk:nsl_single"] = (60*1609)/1000,
["uk:nsl_dual"] = (70*1609)/1000,
["uk:motorway"] = (70*1609)/1000,
["nl:rural"] = 80,
["nl:trunk"] = 100,
["none"] = 140
}
}
2016-04-22 05:31:46 -04:00
end
2017-07-19 09:32:42 -04:00
function process_node(profile, node, result)
2014-03-27 16:19:26 -04:00
-- parse access and barrier tags
local access = find_access_tag(node, profile.access_tags_hierarchy)
if access then
if profile.access_tag_blacklist[access] and not profile.restricted_access_tag_list[access] then
2014-08-27 10:44:40 -04:00
result.barrier = true
2014-03-27 16:19:26 -04:00
end
2014-08-27 10:44:40 -04:00
else
local barrier = node:get_value_by_key("barrier")
if barrier then
-- make an exception for rising bollard barriers
local bollard = node:get_value_by_key("bollard")
local rising_bollard = bollard and "rising" == bollard
if not profile.barrier_whitelist[barrier] and not rising_bollard then
2014-08-27 10:44:40 -04:00
result.barrier = true
end
2014-03-27 16:19:26 -04:00
end
end
2014-08-27 10:44:40 -04:00
-- check if node is a traffic light
local tag = node:get_value_by_key("highway")
if "traffic_signals" == tag then
result.traffic_lights = true
2014-08-27 10:44:40 -04:00
end
end
2017-05-18 08:27:28 -04:00
function process_way(profile, way, result)
-- the intial filtering of ways based on presence of tags
-- affects processing times significantly, because all ways
-- have to be checked.
-- to increase performance, prefetching and intial tag check
-- is done in directly instead of via a handler.
2016-11-17 07:56:17 -05:00
-- in general we should try to abort as soon as
2016-11-17 07:56:17 -05:00
-- possible if the way is not routable, to avoid doing
-- unnecessary work. this implies we should check things that
2017-02-01 14:35:29 -05:00
-- commonly forbids access early, and handle edge cases later.
-- data table for storing intermediate values during processing
local data = {
-- prefetch tags
highway = way:get_value_by_key('highway'),
bridge = way:get_value_by_key('bridge'),
route = way:get_value_by_key('route')
}
-- perform an quick initial check and abort if the way is
-- obviously not routable.
-- highway or route tags must be in data table, bridge is optional
if (not data.highway or data.highway == '') and
(not data.route or data.route == '')
then
return
end
2016-11-17 07:56:17 -05:00
handlers = Sequence {
-- set the default mode for this profile. if can be changed later
-- in case it turns we're e.g. on a ferry
2017-05-18 08:27:28 -04:00
WayHandlers.default_mode,
2016-11-17 07:56:17 -05:00
-- check various tags that could indicate that the way is not
-- routable. this includes things like status=impassable,
-- toll=yes and oneway=reversible
2017-05-18 08:27:28 -04:00
WayHandlers.blocked_ways,
2016-11-17 07:56:17 -05:00
-- determine access status by checking our hierarchy of
-- access tags, e.g: motorcar, motor_vehicle, vehicle
2017-05-18 08:27:28 -04:00
WayHandlers.access,
2016-11-17 07:56:17 -05:00
-- check whether forward/backward directions are routable
2017-05-18 08:27:28 -04:00
WayHandlers.oneway,
-- check a road's destination
2017-05-18 08:27:28 -04:00
WayHandlers.destinations,
2016-11-17 07:56:17 -05:00
-- check whether we're using a special transport mode
2017-05-18 08:27:28 -04:00
WayHandlers.ferries,
WayHandlers.movables,
2016-11-17 07:56:17 -05:00
-- handle service road restrictions
2017-05-18 08:27:28 -04:00
WayHandlers.service,
2016-11-17 07:56:17 -05:00
-- handle hov
2017-05-18 08:27:28 -04:00
WayHandlers.hov,
2016-11-17 07:56:17 -05:00
-- compute speed taking into account way type, maxspeed tags, etc.
2017-05-18 08:27:28 -04:00
WayHandlers.speed,
WayHandlers.surface,
WayHandlers.maxspeed,
WayHandlers.penalties,
2016-11-17 07:56:17 -05:00
-- compute class labels
2017-05-18 08:27:28 -04:00
WayHandlers.classes,
-- handle turn lanes and road classification, used for guidance
2017-05-18 08:27:28 -04:00
WayHandlers.turn_lanes,
WayHandlers.classification,
2016-11-17 07:56:17 -05:00
-- handle various other flags
2017-05-18 08:27:28 -04:00
WayHandlers.roundabouts,
WayHandlers.startpoint,
2016-11-17 07:56:17 -05:00
-- set name, ref and pronunciation
2017-05-18 08:27:28 -04:00
WayHandlers.names,
2017-02-01 14:35:29 -05:00
-- set weight properties of the way
2017-05-18 08:27:28 -04:00
WayHandlers.weights
}
2017-02-01 14:35:29 -05:00
2017-05-18 08:27:28 -04:00
WayHandlers.run(profile,way,result,data,handlers)
2012-09-27 05:35:56 -04:00
end
2012-11-03 03:19:01 -04:00
2017-07-19 09:32:42 -04:00
function process_turn(profile, turn)
-- Use a sigmoid function to return a penalty that maxes out at turn_penalty
-- over the space of 0-180 degrees. Values here were chosen by fitting
-- the function to some turn penalty samples from real driving.
local turn_penalty = profile.turn_penalty
local turn_bias = profile.turn_bias
2017-02-01 14:35:29 -05:00
if turn.has_traffic_light then
2017-05-18 08:27:28 -04:00
turn.duration = profile.properties.traffic_light_penalty
end
if turn.turn_type ~= turn_type.no_turn then
if turn.angle >= 0 then
turn.duration = turn.duration + turn_penalty / (1 + math.exp( -((13 / turn_bias) * turn.angle/180 - 6.5*turn_bias)))
else
turn.duration = turn.duration + turn_penalty / (1 + math.exp( -((13 * turn_bias) * -turn.angle/180 - 6.5/turn_bias)))
end
if turn.direction_modifier == direction_modifier.u_turn then
2017-05-18 08:27:28 -04:00
turn.duration = turn.duration + profile.properties.u_turn_penalty
end
end
-- for distance based routing we don't want to have penalties based on turn angle
2017-05-18 08:27:28 -04:00
if profile.properties.weight_name == 'distance' then
turn.weight = 0
else
turn.weight = turn.duration
2017-03-01 13:50:53 -05:00
end
2017-07-19 09:32:42 -04:00
2017-05-18 08:27:28 -04:00
if profile.properties.weight_name == 'routability' then
2017-03-01 13:50:53 -05:00
-- penalize turns from non-local access only segments onto local access only tags
if not turn.source_restricted and turn.target_restricted then
2017-05-18 08:27:28 -04:00
turn.weight = constants.max_turn_weight
2017-03-01 13:50:53 -05:00
end
end
end
2017-05-18 08:27:28 -04:00
return {
setup = setup,
process_way = process_way,
process_node = process_node,
process_turn = process_turn
}