Model predictive control for autonomous driving

Published:

Model Predictive Control for autonomous driving

This post explains model predictive control formulation for longitudinal and lateral control of the self driving car.

Model Predictive Control

Model predictive control (MPC) is an advanced method of process model that is used to control a process while satisfying a set of constraints. – wikidepia

Error_MPC

There are many advantages of MPC

  • Straight-forward design procedure. Given a model of the system, including constraints, one only needs to set up an objective function that incorporates the control objectives.
  • System latency can be considered easily
  • if the references are know in advance, they can be used in order to optimize the reference tracking
  • It can be used for MIMO and SISO systems.

Vehicle model

There are two kinds famous model for the vehicle modeling

  • Kinematic model
  • Dynamic model

Kinematic models are simplifications of dynamic models that ignore tire forces, gravity, and mass. This simplification reduces the accuracy of the models, but it also makes them more tractable.

Dynamic models encompass tire forces, longitudinal and lateral forces, inertia, gravity, air resistance, drag, mass, and the geometry of the vehicle. Therefore, it is more accurate than the kinematic model.

So, which model is more suitable for using MPC for autonomous driving control?

In this paper, thankfully, the kinematic model and dynamic model were analyzed from the perspective of MPC application in autonomous vehicles. As a result, it was shown that the performance difference between the kinematic and the dynamic model was not significant at 200ms discretization time.

Therefore, I chosen the kinematic model because the dynamic model need tire coefficient which is hard to estimate.

Kinematic model equation:

\[x_{t+1} = x_{t}+v_{t}*cos(\psi_{t})*dt\\ y_{t+1} = y_{t}+v_{t}*sin(\psi_{t})*dt\\ \psi_{t+1} = \psi_{t}+{v_{t}\over{L_{f}}}*\delta_{t}*dt\\ v_{t+1} = v_{t}+a_{t}*dt\\\]

If you want to deeply understanding the vehicle models, I highly recommend this book which is written by Rajamani.

Error states

I will use three error types for the MPC.

Error_MPC

  • Cross track error (CTE)

    • Express the error between the reference path and the vehicle’s position.
    • $cte_{t+1}=cte_{t}+v_t*sin(e\psi_t) *dt$
    • In vehicle coordinate, If we assume the reference path is $f(x_t)$, then cte is $cte_{t} = f(x_t)-y_t$.
    • Finally, we can express the CTE as $cte_{t+1}=f(x_{t})-y_{t}+v_t*sin(e\psi_t) *dt$
  • Heading error (EPSI)

    • Express the heading error between the reference path and the vehicle’s heading angle
    • $e\psi_{t+1}=e\psi_t+{v_{t}\over{L_{f}}} *\delta_{t} *dt$
    • Error $\psi$ means $e\psi_t=\psi_t-\psi des_t$
    • Therefore, $e\psi_{t+1}=\psi_t-\psi des_{t}+{v_{t}\over{L_{f}}} *\delta_{t} *dt$
  • Speed error

    • Express the error between reference speed and the vehicle’s speed
\[e_t=\left(\begin{array}{ll1} {cte_t} \\ {epsi_t} \\ {v_{err,t}} \\ \end{array}\right)\]

MPC formulation

Then, MPC can be formulated:

\[\min_u\hspace{0.2cm}J=\Sigma_{t=0}^{N}e_t^T Qe_t +\Sigma_{t=0}^{N-1}u_t^T R_1u_t+\Sigma_{t=0}^{N-2}\Delta u_t^T R_2\Delta u_t\\ s.t \hspace{1cm}X_{t+1}=f(X_t,u_t)\hspace{3cm}t= 0,1,\dots,N\\ \hspace{2cm}u_{min}<u_t<u_{max}\hspace{3cm}t= 0,1,\dots,N\\\]

$N$ is prediction horizon

MPC will optimize inputs that minimize error states, the magnitude of inputs and derivative of inputs. The inputs are steering angle and throttle value.

MPC implementation by using the IPOPT library

Now, let’s implement MPC by ipopt library. Ipopt (Interior Point Optimizer, pronounced “Eye-Pea-Opt”) is an open source software package for large-scale nonlinear optimization. In order to use ipopt for the optimization, we also need the differentiation library. In this post, CppAD will be used for it. This link is well documented for the ipopt with CppAD.

In ipopt, cost function is in variable fg[0] . In above MPC formulation it is $J$

    for (int t = 0; t < N; t++){
      
      // error states
      // cte (cross-track error)
      fg[0] += w_cte*CppAD::pow(vars[cte_start + t], 2);
      // epsi (heading error)
      fg[0] += w_epsi*CppAD::pow(vars[epsi_start + t], 2);
      // speed error
      fg[0] += w_verr*CppAD::pow(vars[v_start + t] - desired_speed, 2);  
      
      // inputs
      if (t < N - 1) {
        // steering
        fg[0] += w_delta*CppAD::pow(vars[delta_start + t], 2);
        // acceleration
        fg[0] += w_a*CppAD::pow(vars[a_start + t], 2);
      }
      // delta inputs
      if (t < N - 2) {
        //steering
        fg[0] += w_d_delta*CppAD::pow(vars[delta_start + t + 1] - vars[delta_start + t], 2);
        //acceleration
        fg[0] += w_d_a*CppAD::pow(vars[a_start + t + 1] - vars[a_start + t], 2);    
      }
    }

Then, let’s make equality condition. In this project, 3-order polynomial trajectory was chosen as reference trajectory

       // 3rd order polynomial
      AD<double> f0 = coeffs[0] + coeffs[1]*x0 + coeffs[2]*pow(x0,2) + coeffs[3]*pow(x0,3);
      AD<double> psides0 = CppAD::atan(coeffs[1] + 2*coeffs[2]*x0 + 3*coeffs[3]*pow(x0,2));
      // Setup the rest of the model constraints
      fg[1 + x_start + t]    = x1    - (x0 + v0 * CppAD::cos(psi0) * dt);
      fg[1 + y_start + t]    = y1    - (y0 + v0 * CppAD::sin(psi0) * dt);
      fg[1 + psi_start + t]  = psi1  - (psi0 + v0 * delta0 / Lf * dt);
      fg[1 + v_start + t]    = v1    - (v0 + a0 * dt);
      fg[1 + cte_start + t]  = cte1  - ((f0 - y0) + (v0 * CppAD::sin(epsi0) * dt));
      fg[1 + epsi_start + t] = epsi1 - ((psi0 - psides0) + v0 * delta0 / Lf * dt);  

Also, inequality condition.

  // The upper and lower limits of delta are set to -25 and 25
  // degrees (values in radians).
  for (int i = delta_start; i < a_start; i++) {
    // limits set at +/- 0.4363 (radian) to represent 25 degrees
    vars_lowerbound[i] = -0.4363;
    vars_upperbound[i] =  0.4363;
  }

  // Acceleration/decceleration upper and lower limits.
  for (int i = a_start; i < n_vars; i++) {
    vars_lowerbound[i] = -1;
    vars_upperbound[i] =  1;
  } 
  

Simulation

  • green line: MPC optimal trajectory
  • Yellow line: Reference trajectory

If you would like to see overall codes, Please visit my repo