Skip to content

Prediction Predictor Manager

Yuesong Xie edited this page Apr 4, 2019 · 1 revision

PredictorManager管理器

ContainerManager管理器得到了每个障碍物的LaneGraph,里面包含了若干LaneSequence,每个LaneSequence指代障碍物运动的轨迹,或者叫做运动方案。而一个LaneSequence包含若干LaneSegment,这些LaneSegment就是该LaneSequence的运动车道,或者叫做路段。

EvaluatorManager管理器对每个障碍物的每条运动路线(LaneSequence)进行评估,评估指标共62项,包括22项障碍物特征,40项车道特征。最后得到每个LaneSequence的概率。

当经过EvaluatorManager评估以后,就可以根据规则得到障碍物有可能的运动轨迹(运动方案),这时候我们就可以针对每个障碍物生成一条短时间内的运动轨迹。这些运动轨迹可以交由Planning模块进行路径重规划。这里的障碍物短时间内运动轨迹是什么概念?障碍物当前位置(x,y),还有速度,加速度,速度方向。由ContainerManager可知,我们对LaneSegment是经过离散采样保存的,所以障碍物短时间内的运动轨迹是指:当前位置到LaneSequence的第一个最近的LanePoint的运动轨迹。如果一个障碍物的LaneGraph由N条LaneSequence,那么该障碍物就对应了m条轨迹,这m条轨迹是概率最大的那几条(具体挑选规则在下文提到)

/// file in apollo/modules/prediction/conf/prediction_conf.pb.txt
obstacle_conf {
  obstacle_type: VEHICLE
  obstacle_status: ON_LANE
  evaluator_type: MLP_EVALUATOR
  predictor_type: MOVE_SEQUENCE_PREDICTOR
}
obstacle_conf {
  obstacle_type: VEHICLE
  obstacle_status: OFF_LANE
  predictor_type: FREE_MOVE_PREDICTOR
}
obstacle_conf {
  obstacle_type: PEDESTRIAN
  predictor_type: FREE_MOVE_PREDICTOR
}
obstacle_conf {
  obstacle_type: BICYCLE 
  obstacle_status: ON_LANE
  evaluator_type: MLP_EVALUATOR
  predictor_type: MOVE_SEQUENCE_PREDICTOR
}
obstacle_conf {
  obstacle_type: BICYCLE
  obstacle_status: OFF_LANE
  predictor_type: FREE_MOVE_PREDICTOR
}
obstacle_conf {
  obstacle_type: UNKNOWN
  obstacle_status: ON_LANE
  evaluator_type: MLP_EVALUATOR
  predictor_type: LANE_SEQUENCE_PREDICTOR
}
obstacle_conf {
  obstacle_type: UNKNOWN
  obstacle_status: OFF_LANE
  predictor_type: FREE_MOVE_PREDICTOR
}

还是看一下预测器的多累,上面看到一共用到3类预测器:

Lane sequence(LANE_SEQUENCE_PREDICTOR): 沿着车道(中心线)行驶,比较简单地方法 Move sequence(MOVE_SEQUENCE_PREDICTOR): 根据其运动模式,形式向车道中心线后沿着车道行驶 Free movement(FREE_MOVE_PREDICTOR):自由移动

/// file in apollo/modules/prediction/predictor/predictor_manager.cc
void PredictorManager::Run(const PerceptionObstacles& perception_obstacles) {
  for (const auto& perception_obstacle : perception_obstacles.perception_obstacle()) {
   
    PredictionObstacle prediction_obstacle;
    prediction_obstacle.set_timestamp(perception_obstacle.timestamp());
    Obstacle* obstacle = obstacles_container->GetObstacle(id);
    if (obstacle != nullptr) {
      // Step 1. get predictor
      // Step 2. predict
      if (predictor != nullptr) {
        predictor->Predict(obstacle);                                 // 计算每个障碍物的短时间运动轨迹
        for (const auto& trajectory : predictor->trajectories()) {
          prediction_obstacle.add_trajectory()->CopyFrom(trajectory); // 将轨迹加入prediction_obstacle
        }
      }
    }
    prediction_obstacle.set_predicted_period(FLAGS_prediction_duration);
    prediction_obstacle.mutable_perception_obstacle()->CopyFrom(perception_obstacle);

    prediction_obstacles_.add_prediction_obstacle()->CopyFrom(prediction_obstacle); // 将prediction_obstacle加入到向量中,即保存该障碍物的所有可能轨迹
  }
}

从上面代码很容易看到预测阶段对每个障碍物分别预测其轨迹,而且一个障碍物的轨迹不止一条(有些LaneSequence概率都比较大)。接下来我们看一下predictor->Predict(obstacle);这个过程,我们选取MOVE_SEQUENCE_PREDICTOR模式进行分析。

/// file in apollo/modules/prediction/predictor/move_sequence/move_sequence_predictor.cc
void MoveSequencePredictor::Predict(Obstacle* obstacle) {
  const Feature& feature = obstacle->latest_feature();
  // Step 1. 对该障碍物LaneGraph中每个LaneSequence根据其evaluator的概率进行预测,是否和规划好的轨迹重叠
  int num_lane_sequence = feature.lane().lane_graph().lane_sequence_size();
  std::vector<bool> enable_lane_sequence(num_lane_sequence, true);
  FilterLaneSequences(feature, lane_id, &enable_lane_sequence);
  for (int i = 0; i < num_lane_sequence; ++i) {
    const LaneSequence& sequence = feature.lane().lane_graph().lane_sequence(i);

    // Step 2. 画出短时间内障碍物运动的轨迹
    std::vector<TrajectoryPoint> points;
    DrawMoveSequenceTrajectoryPoints(*obstacle, sequence, FLAGS_prediction_duration, FLAGS_prediction_period, &points);
    // Step 3. 生成轨迹并保存
    Trajectory trajectory = GenerateTrajectory(points);
    trajectory.set_probability(sequence.probability());
    trajectories_.push_back(std::move(trajectory));
  }
}

整个过程相对来说比较简单,但是每个过程有一些难点,这里我们简单地可以将Predict函数分为三个过程:

  1. 对各个LaneSequence根据其上阶段evalute的结果(每个运动方案的概率)进行筛选,首先障碍物是智能体,他运动的轨迹也要参考别的障碍物,如果他运动的轨迹和主车当前位置过近,那么是很危险的,所以正常思维判断,这个LaneSequence是不可行的。举个例子,主车在车道1上行驶,障碍物在车道2行驶,两车相距2米,下时刻突然障碍物变道到主车的车道1是不太可能的。

  2. 对那么不会造成相撞的障碍物运动方案进行短时间运动轨迹的生成,生成思路很简单,插值。

  3. 轨迹保存,这个就比较简单DrawMoveSequenceTrajectoryPoints函数画出了一系列的运动点,只要把它封装成轨迹点后放入Trajectory类即可。

LaneSequence过滤--FilterLaneSequences

这里我们简单地看一下步骤1和步骤2的实现方案。步骤1 LaneSequence的筛选预过滤步骤如下:

  1. 计算每个LaneSequence对应障碍物的运动,(保持前进、左转还是右转)。

计算当前位置的车道: lane_id (feature.lane().lane_feature().lane_id()) LaneSequence第一个LaneSegment对应的车道线: first_lane_id(feature.lane().lane_graph().lane_sequence(i).lane_segment(0).lane_id())

如果first_lane_id == lane_id,则障碍物保持前进;如果first_lane_id在lane_id左侧,那么障碍物左转;如果first_lane_id在lane_id右侧,那么障碍物右转

  1. 计算ADCTrajectory到LaneSequence的距离

分一下几种情况

  • 如果LaneSequence和ADCTrajectory没有重叠,那么返回浮点数最大值即可

  • 如果LaneSequence和ADCTrajectory存在重叠,那么计算ADCTrajectory第一个点p1(也就是主车的位置)到LaneSequence的第一个LanePoint(准确的说是p1之后的LaneSequence中第一个LanePoint)之间的距离,计算方法也比较简单,将ADCTrajectory第一个点投影到LaneSegment对应的车道上,返回两个点的累计距离差 renturn fabs(p1.s-p2.s)

  1. 最后对上述距离做是否有效的统计

如果距离小于一个阈值(FLAGS_lane_change_dist,默认10米),那么作为智能体障碍物的LaneSequence,正常情况下是不会选择这条行驶路线的

  1. 计算上述有效LaneSequence的最大概率以及转弯最大概率

这个过程比较简单,在代码中可以清晰地看出。至于为什么要额外计算转弯的最大概率,一种可能的原因是evaluate评估得到的结果更多的是偏向于直行,也就是那么多LaneSequence中直行的概率是很大的,而转弯的概率比较小,如果直接根据最大概率一锤定音,那么很难出现转弯的情况。如果把转弯单独拿出来,这会更加障碍物的评估准确率。最终得到的有效LaneSequence可能有多条,所以每个障碍物对应的Trajectory可能有多条。

轨迹曲线生成--DrawMoveSequenceTrajectoryPoints

Move Sequence Predictor

在Move Sequence中,轨迹曲线的生成比较繁琐,主要是分别建立主方向和侧方向的多项式函数,然后每个时刻在多项式上插值就可以得到采样点的累积距离s和侧方相对距离l主要的计算过程分为:

  1. 计算障碍物到车道中心线的时间
Eigen::Vector2d position(feature.position().x(), feature.position().y());
double time_to_lat_end_state = std::max(FLAGS_default_time_to_lat_end_state, ComputeTimeToLatEndConditionByVelocity(obstacle, lane_sequence));

这个相对来说不难理解,因为障碍物当前位置由x和y方向的速度vx和vy,同时可以计算得到障碍物到LaneSequence的之后最近LanePoint所在Lane的侧面距离relative_l,已经LaneSequence的第一个最近LanePoint的方向lane_heading,那么只要计算vx和vy在lane_heading垂直方向的分量v_l = v_y * std::cos(lane_heading) - v_x * std::sin(lane_heading),然后relative_l / v_l即可。

  1. 计算lane_heading侧方向(Lateral)的多项式表达式

在侧方向上代码中构建了5次多项式来拟合这个曲线:

$$ lateral = f(t) = a_0 + a_1t + a_2t^2 + a_3t^3 + a_4t^4 + a_5t^5 $$

计算多项式函数第一个点信息,也就是障碍物当前位置信息:

  • 与原点距离,也就是f(0),也是代码中的l0
  • 侧方轴速度,也就是f'(0),也是代码中的dl0
  • 侧方轴加速度,也就是f''(0),也是代码中的ddl0
double l0 = (cross_prod > 0) ? shift : -shift;
double dl0 = v * std::sin(theta - start_lane_point.heading());
double ddl0 = a * std::sin(theta - start_lane_point.heading());

经过了ts时间以后,障碍物就到了第二个点,也就是LaneSequence的第一个LanePoint,他的信息为:

  • 与原点距离,也就是f(ts),为0,也是代码中的l1
  • 侧方轴速度,也就是f'(ts),为0,也是代码中的dl1
  • 侧方轴加速度,也就是f''(ts),为0,也是代码中的dl1

为什么为0?因为多项式坐标系是以LanePoint为原点,侧方轴为x轴。

所以上述求解dl0和ddl0过程中,需要减去LanePoint的方向,将heading转化到新的坐标系中。乘以sin是计算侧方轴上的分量。

那么就有:

$$ f(0) = a_0 = l0 $$

$$ f'(0) = a_1 = dl0 $$

$$ f''(0) = 2a_2 = ddl0 $$

所以5次多项式的前面三个系数为: l0, dl0, 0.5*ddl0,对应代码中的

coefficients->operator[](0) = l0;
coefficients->operator[](1) = dl0;
coefficients->operator[](2) = ddl0 / 2.0;

现在将第二个点信息带入得到:

$$ f(t_s) = l0 + dl0·t_s + 0.5·ddl0·{t_s}^2 + a_3{t_s}^3 + a_4{t_s}^4 + a_5{t_s}^5 = l1 $$

$$ f'(t_s) = dl0 + ddl0·t_s + 3a_3{t_s}^2 + 4a_4{t_s}^3 + 5a_5{t_s}^4 = dl1 $$

$$ f''(t_s) = ddl0 + 6a_3t_s + 12a_4{t_s}^2 + 20a_5{t_s}^3 = ddl1 $$

通过移项可以得到:

$$ c0 = a_3 + a_4t_s + a_5{t_s}^2 = (l1 - 0.5·ddl0·{t_s}^2 - dl0·t_s - l0) / {t_s}^3 $$

$$ c1 = 3a_3 + 4a_4t_s + 5a_5{t_s}^2 = (dl1 - ddl0·t_s - dl0) / {t_s}^2 $$

$$ c2 = 6a_3 + 12a_4t_s + 20a_5{t_s}^2 = (ddl1 - ddl0) / t_s $$

以上三个一元二次多项式就可以求解了,最后的结果为:

$$ a_3 = 0.5 * (20.0 * c0 - 8.0 * c1 + c2) $$

$$ a_4 = (-15.0 * c0 + 7.0 * c1 - c2) / t_s $$

$$ a_5 = (6.0 * c0 - 3.0 * c1 + 0.5 * c2) / {t_s}^2 $$

这也是对应代码:

double p = time_to_end_state;
double p2 = p * p;
double p3 = p2 * p;
double c0 = (l1 - 0.5 * p2 * ddl0 - dl0 * p - l0) / p3;
double c1 = (dl1 - ddl0 * p - dl0) / p2;
double c2 = (ddl1 - ddl0) / p;

coefficients->operator[](3) = 0.5 * (20.0 * c0 - 8.0 * c1 + c2);
coefficients->operator[](4) = (-15.0 * c0 + 7.0 * c1 - c2) / p;
coefficients->operator[](5) = (6.0 * c0 - 3.0 * c1 + 0.5 * c2) / p2;

这样多项式拟合和求解就玩完成了。

  1. 计算lane_heading主方向(Longitudinal)的多项式表达式

具体的计算方法与2中一致,但是Longitudinal方向构建的是4次多项式,仅此区别。

  1. 最后一步就是根据两个多项式进行插值,可以看到Apollo对这段短时间的轨迹一共插值50个点total_num = static_cast<size_t>(total_time / period),并且分装成PathPoint
  for (size_t i = 0; i < total_num; ++i) {
	TrajectoryPoint trajectory_point;
	PathPoint path_point;
	path_point.set_x(point.x());
	path_point.set_y(point.y());
	path_point.set_z(0.0);
	path_point.set_theta(theta);
	path_point.set_lane_id(lane_id);
	trajectory_point.mutable_path_point()->CopyFrom(path_point);
	trajectory_point.set_v(lane_speed);
	trajectory_point.set_a(lane_acc);
	trajectory_point.set_relative_time(relative_time);
	points->emplace_back(std::move(trajectory_point));

	while (lane_s > PredictionMap::LaneById(lane_id)->total_length() &&
           lane_segment_index + 1 < lane_sequence.lane_segment_size()) {
      lane_segment_index += 1;
      lane_s = lane_s - PredictionMap::LaneById(lane_id)->total_length();
      lane_id = lane_sequence.lane_segment(lane_segment_index).lane_id();
    }
  }

代码最后的while作用就是针对多LaneSegment的LaneSequence进行变道处理。

Lane Sequence Predictor

相比Move Sequence类型的预测器,Lane Sequence预测的DrawMoveSequenceTrajectoryPoints函数就比较简单,我们额外看一下他的轨迹计算方法。Lane Sequence预测策略是假设障碍物会沿着车道(中心线)运动,那么短时间内从当前位置P(x,y)到LaneSequence的第一个LanePoint(x',y')的运动轨迹,可以参考下图

img

计算方法为:

  1. 首先将当前位置坐标P(x,y)投影到LaneSequence的第一个最近的LanePoint对应的车道中心线上,得到对应车道上的点,此时车道上的投影点有累计距离lane_s和对应的投影距离lane_l,这里需要注意lane_l(对应上图中的realitive_l)是有正负(方向)的,若P2在P1下方,lane_l为负数;否则在上方为正数。

  2. 然后根据上述的period和speed可以计算每两次采样前进的距离,lane_s+=period*speed

  3. 最后将车道中心线上的lane_s对应的点根据lane_l投影到真实障碍物运动路径上(左图中的虚线)

Free Move Predictor

Free Move预测模式更加简单,仅仅是根据障碍物的运动模式进行预测。举个例子

  • 障碍物当前时刻坐标: P(x,y)
  • 障碍物当前时刻速度(矢量): (v_x, v_y)
  • 障碍物当前时刻速度(标量): speed=sqrt(v_x*v_x+v_y*v_y)
  • 障碍物当前时刻加速度(矢量): (acc_x, acc_y)
  • 障碍物当前时刻加速度: acc
  • 障碍物当前时刻偏航角: theta(因为不知道上时刻的障碍物位置,所以不知道偏航角,计算第一个点时时暂时保留)

当经过period时刻以后:

  • 障碍物x方向运动的距离:s_x = v_x * period + 0.5 * acc_x * period * period

  • 障碍物x方向运动的距离:s_y = v_y * period + 0.5 * acc_y * period * period

  • 障碍物坐标: P2(x+s_x, y+s_y)

  • 障碍物速度(矢量): (v_x+acc_x*period, v_y+acc_y*period)

  • 障碍物当前时刻速度(标量): speed_t=sqrt(v_x*v_x+v_y*v_y)

  • 障碍物当前时刻加速度(矢量): (acc_x, acc_y) (假设这个短时间内,加速度不变)

  • 障碍物当前时刻加速度: acc = (speed_t - speed) / period

  • 上时刻障碍物偏航角: theta = actan(P2.y-P.y, P2.x-P.x) (利用本次位置和上一次位置,可以计算上一次的偏航角)

最终总结Predictor的作用:根据LaneSequence的概率,以及预先规划好的ADCTrajectory对LaneSequence进行筛选,去掉那些不合理的行驶方案,最后得到一些短时间轨迹的集合。

从参数FLAGS_prediction_duration(默认5.0s)和FLAGS_prediction_period(默认0.1s),可以看到,Prediction模块其实是对障碍物未来5s内的运动状态进行预测,每0.1s进行一次位置的采样。

Clone this wiki locally