123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644 |
- #include "frenet_planner.h"
- #include <adc_tools/transfer.h>
- #include <common/car_status.h>
- #include <adc_tools/parameter_status.h>
- #include <adc_planner/lane_change_planner.h>
- #include<Eigen/Dense>
- using namespace std;
- using namespace Eigen;
- iv::decition::FrenetPlanner::FrenetPlanner(){
- this->planner_id = 1;
- this->planner_name = "Frenet";
- this->lidarGridPtr = NULL;
- }
- iv::decition::FrenetPlanner::~FrenetPlanner(){
- free(this->lidarGridPtr);
- }
- std::vector<iv::Point2D> iv::decition::FrenetPlanner::getPath(GPS_INS now_gps_ins, const std::vector<GPSData>& gpsMapLine, int PathPoint, double offset, iv::LidarGridPtr lidarGridPtr){
- this->roadNow = -1;
- this->now_gps_ins = now_gps_ins;
- this->gpsMapLine = gpsMapLine;
- this->PathPoint = PathPoint;
- this->lidarGridPtr = lidarGridPtr;
- LaneChangePlanner *lcp = new LaneChangePlanner();
- std::vector<iv::Point2D> gpsTrace = lcp->getPath(now_gps_ins,gpsMapLine,PathPoint,0,lidarGridPtr);
- delete lcp;
- double realSecSpeed = now_gps_ins.speed / 3.6;
- leftWidth = offset;
- rightWidth = offset;
- std::vector<iv::Point2D> trace = getGpsTrace_AfterCalcFrenetTrace(gpsTrace,realSecSpeed);
- return trace;
- }
- int iv::decition::FrenetPlanner::chooseRoadByFrenet(GPS_INS now_gps_ins, const std::vector<GPSData>& gpsMapLine, int PathPoint, double offsetL, double offsetR, iv::LidarGridPtr lidarGridPtr,int roadNow){
- this->roadNow = roadNow;
- this->aimRoadByFrenet = -1;
- this->now_gps_ins = now_gps_ins;
- this->gpsMapLine = gpsMapLine;
- this->PathPoint = PathPoint;
- this->lidarGridPtr = lidarGridPtr;
- LaneChangePlanner *lcp = new LaneChangePlanner();
- std::vector<iv::Point2D> gpsTrace = lcp->getPath(now_gps_ins,gpsMapLine,PathPoint,0,lidarGridPtr);
- delete lcp;
- double realSecSpeed = now_gps_ins.speed / 3.6;
- leftWidth = offsetL;
- rightWidth = offsetR;
- std::vector<iv::Point2D> trace = getGpsTrace_AfterCalcFrenetTrace(gpsTrace,realSecSpeed);
- return aimRoadByFrenet;
- }
- std::vector<iv::Point2D> iv::decition::FrenetPlanner::getGpsTrace_AfterCalcFrenetTrace(const std::vector<iv::Point2D>& gpsTrace,double realsecSpeed){
- vector<Point2D> trace;
- vector<FrenetPoint> frenet_s;
- double stp=0.0;
-
- FrenetPoint tmp={.x=gpsTrace[0].x,.y=gpsTrace[0].y,.s=0.0,.d=0.0};
- frenet_s.push_back(tmp);
- for(int i=1; i<gpsTrace.size(); ++i){
- stp+=iv::decition::GetDistance(gpsTrace[i-1],gpsTrace[i]);
- tmp={.x=gpsTrace[i].x,.y=gpsTrace[i].y,.s=stp,.d=0.0};
- frenet_s.push_back(tmp);
- }
-
- FrenetPoint car_now_at_frenet = getFrenetfromXY(0,0,gpsTrace,this->gpsMapLine,this->PathPoint,this->now_gps_ins);
- double c_s_speed = realsecSpeed * cos(car_now_at_frenet.tangent_Ang-M_PI/2);
- double c_d_speed = realsecSpeed * sin(car_now_at_frenet.tangent_Ang-M_PI/2);
- double realAcc = ServiceCarStatus.mfCalAcc;
- double c_s_dd = realAcc * cos(car_now_at_frenet.tangent_Ang-M_PI/2);
- double c_d_dd = realAcc * sin(car_now_at_frenet.tangent_Ang-M_PI/2);
- vector<Point2D> path=frenet_optimal_planning(car_now_at_frenet,frenet_s,c_s_speed,c_d_speed,c_d_dd);
- return path;
- }
- vector<iv::Point2D> iv::decition::FrenetPlanner::frenet_optimal_planning(iv::decition::FrenetPoint car_now_frenet_point,
- const std::vector<FrenetPoint>& frenet_s, double c_speed, double c_d_d, double c_d_dd){
- double s0 = car_now_frenet_point.s;
- double c_d = car_now_frenet_point.d;
- vector<iv::Point2D> gpsTrace;
- vector<Frenet_path> fplist=calc_frenet_paths(c_speed, c_d, c_d_d, c_d_dd, s0);
- sort(fplist.begin(),fplist.end(),comp);
- for(int i=0; i<fplist.size(); ++i){
- calc_global_single_path(fplist[i],frenet_s);
- if(check_single_path(fplist[i])){
- gpsTrace = frenet_path_to_gpsTrace(fplist[i]);
- aimRoadByFrenet = fplist[i].roadflag;
- return gpsTrace;
- }
- }
- return gpsTrace;
- }
- vector<iv::decition::Frenet_path> iv::decition::FrenetPlanner::calc_frenet_paths(double c_speed, double c_d, double c_d_d, double c_d_dd, double s0){
- vector<iv::decition::Frenet_path> frenet_paths;
- int roadNum = round(abs((leftWidth-rightWidth)/ServiceParameterStatus.D_ROAD_W));
-
- for(double di=leftWidth; di<=rightWidth; di+=ServiceParameterStatus.D_ROAD_W,roadNum--){
- if(roadNum == this->roadNow)
- continue;
-
- for(double Ti=ServiceParameterStatus.MINT; Ti<ServiceParameterStatus.MAXT; Ti+=ServiceParameterStatus.DT){
- Frenet_path fp;
- fp.roadflag = roadNum;
-
- Quintic_polynomial lat_qp = Quintic_polynomial(c_d, c_d_d, c_d_dd, di, 0.0, 0.0, Ti);
- for(double i=0; i<Ti; i+=ServiceParameterStatus.D_POINT_T)
- fp.t.push_back(i);
- for(int i=0; i<fp.t.size(); ++i){
- fp.d.push_back(lat_qp.calc_point(fp.t[i]));
- fp.d_d.push_back(lat_qp.calc_first_derivative(fp.t[i]));
- fp.d_dd.push_back(lat_qp.calc_second_derivative(fp.t[i]));
- fp.d_ddd.push_back(lat_qp.calc_third_derivative(fp.t[i]));
- }
-
- for(double tv=ServiceParameterStatus.TARGET_SPEED - ServiceParameterStatus.D_T_S * ServiceParameterStatus.N_S_SAMPLE;
- tv<(ServiceParameterStatus.TARGET_SPEED + ServiceParameterStatus.D_T_S * ServiceParameterStatus.N_S_SAMPLE);
- tv+=ServiceParameterStatus.D_T_S){
- Frenet_path tfp = fp;
- Quartic_polynomial lon_qp = Quartic_polynomial(s0, c_speed, 0.0, tv, 0.0, Ti);
- for(int i=0; i<fp.t.size(); ++i){
- tfp.s.push_back(lon_qp.calc_point(fp.t[i]));
- tfp.s_d.push_back(lon_qp.calc_first_derivative(fp.t[i]));
- tfp.s_dd.push_back(lon_qp.calc_second_derivative(fp.t[i]));
- tfp.s_ddd.push_back(lon_qp.calc_third_derivative(fp.t[i]));
- }
-
- double Jp = inner_product(tfp.d_ddd.begin(), tfp.d_ddd.end(), tfp.d_ddd.begin(), 0);
-
- double Js = inner_product(tfp.s_ddd.begin(), tfp.s_ddd.end(), tfp.s_ddd.begin(), 0);
-
- double ds = pow((ServiceParameterStatus.TARGET_SPEED - tfp.s_d.back()),2);
-
- tfp.cd = ServiceParameterStatus.KJ * Jp + ServiceParameterStatus.KT * Ti + ServiceParameterStatus.KD * tfp.d.back() * tfp.d.back();
-
- tfp.cv = ServiceParameterStatus.KJ * Js + ServiceParameterStatus.KT * Ti + ServiceParameterStatus.KD * ds;
-
- tfp.cf = ServiceParameterStatus.KLAT * tfp.cd + ServiceParameterStatus.KLON * tfp.cv;
- frenet_paths.push_back(tfp);
- }
- }
- }
- return frenet_paths;
- }
- void iv::decition::FrenetPlanner::calc_global_paths(vector<Frenet_path>& fplist,const std::vector<FrenetPoint>& frenet_s){
- for(auto& fp:fplist){
- for(int i=0; i<fp.s.size(); ++i){
- double ix,iy;
- getXYfromFrenet(&ix,&iy,fp.s[i],fp.d[i],frenet_s,this->gpsMapLine,this->PathPoint,this->now_gps_ins);
- fp.x.push_back(ix);
- fp.y.push_back(iy);
- }
- for(int i=0; i<(fp.x.size()-1); ++i){
- double dx = fp.x[i+1] - fp.x[i];
- double dy = fp.y[i+1] - fp.y[i];
- fp.yaw.push_back(atan2(dy,dx));
- fp.ds.push_back(sqrt(dx*dx + dy*dy));
- }
- fp.ds.push_back(fp.ds.back());
- fp.yaw.push_back(fp.yaw.back());
- for(int i = 0; i < (fp.yaw.size() - 1); ++i){
- fp.c.push_back((fp.yaw[i + 1] - fp.yaw[i]) / fp.ds[i]);
- }
- }
- }
- void iv::decition::FrenetPlanner::calc_global_single_path(Frenet_path &fp, const std::vector<FrenetPoint> &frenet_s){
- for(int i=0; i<fp.s.size(); ++i){
- double ix,iy;
-
- getXYfromFrenet(&ix,&iy,fp.s[i],fp.d[i],frenet_s,this->gpsMapLine,this->PathPoint,this->now_gps_ins);
- fp.x.push_back(ix);
- fp.y.push_back(iy);
- }
- for(int i=0; i<(fp.x.size()-1); ++i){
- double dx = fp.x[i+1] - fp.x[i];
- double dy = fp.y[i+1] - fp.y[i];
- fp.yaw.push_back(atan2(dy,dx));
- fp.ds.push_back(sqrt(dx*dx + dy*dy));
- }
- fp.ds.push_back(fp.ds.back());
- fp.yaw.push_back(fp.yaw.back());
- for(int i = 0; i < (fp.yaw.size() - 1); ++i){
- fp.c.push_back((fp.yaw[i + 1] - fp.yaw[i]) / fp.ds[i]);
- }
- }
- vector<iv::decition::Frenet_path> iv::decition::FrenetPlanner::check_paths(const vector<Frenet_path>& fplist){
- vector<Frenet_path> okind;
- int i=0;
- int j=0;
- for(i=0; i<fplist.size(); ++i){
- cout<<"&&&&&&&&&&fplist[i].roadflag: "<<fplist[i].roadflag<<endl;
- for(j=0; j<fplist[i].s_d.size(); ++j){
- if(fplist[i].s_d[j]>ServiceParameterStatus.MAX_SPEED)
- goto ContinueFlag;
- }
- for(j=0; j<fplist[i].s_dd.size(); ++j){
- if(abs(fplist[i].s_dd[j])>ServiceParameterStatus.MAX_ACCEL)
- goto ContinueFlag;
- }
- for(j=30; j<fplist[i].c.size()-30; ++j){
- if(abs(fplist[i].c[j])>ServiceParameterStatus.MAX_CURVATURE)
- goto ContinueFlag;
- }
- if(!check_collision(fplist[i]))
- continue;
- okind.push_back(fplist[i]);
- ContinueFlag:continue;
- }
- return okind;
- }
- bool iv::decition::FrenetPlanner::check_single_path(const iv::decition::Frenet_path &fp){
- int j=0;
- cout<<"&&&&&&&&&&fp.roadflag: "<<fp.roadflag<<endl;
- for(j=0; j<fp.s_d.size(); ++j){
- if(fp.s_d[j]>ServiceParameterStatus.MAX_SPEED)
- return false;
- }
- for(j=0; j<fp.s_dd.size(); ++j){
- if(abs(fp.s_dd[j])>ServiceParameterStatus.MAX_ACCEL)
- return false;
- }
- for(j=30; j<fp.c.size()-30; ++j){
- if(abs(fp.c[j])>ServiceParameterStatus.MAX_CURVATURE)
- return false;
- }
- if(!check_collision(fp))
- return false;
- return true;
- }
- bool iv::decition::FrenetPlanner::check_collision(const iv::decition::Frenet_path &frenet_path){
- std::vector<Point2D> gpsTrace = frenet_path_to_gpsTrace(frenet_path);
- double obs=-1;
-
- if(obs > 0 && obs < 30)
- return false;
- else
- return true;
- }
- vector<iv::Point2D> iv::decition::FrenetPlanner::frenet_path_to_gpsTrace(const Frenet_path& frenet_path){
- vector<iv::Point2D> gpsTrace;
- for (int i=0; i<frenet_path.x.size(); ++i) {
- gpsTrace.push_back(Point2D(frenet_path.x[i],frenet_path.y[i]));
- gpsTrace[i].speed = sqrt(frenet_path.d_d[i]*frenet_path.d_d[i]+frenet_path.s_d[i]*frenet_path.s_d[i])*3.6;
- }
- return gpsTrace;
- }
- double iv::decition::FrenetPlanner::distance(double x1, double y1, double x2, double y2)
- {
- return sqrt((x2-x1)*(x2-x1)+(y2-y1)*(y2-y1));
- }
- int iv::decition::FrenetPlanner::ClosestWaypoint(double x, double y, const std::vector<Point2D>& gpsTrace)
- {
- double closestLen = 100000;
- int closestWaypoint = 0;
- for(int i = 1; i < gpsTrace.size(); i++)
- {
- double map_x = gpsTrace[i].x;
- double map_y = gpsTrace[i].y;
- double dist = distance(x,y,map_x,map_y);
- if(dist < closestLen)
- {
- closestLen = dist;
- closestWaypoint = i;
- }
- }
- return closestWaypoint;
- }
-
-
-
- iv::decition::FrenetPoint iv::decition::FrenetPlanner::XY2Frenet(double x, double y, const std::vector<iv::Point2D>& gpsTrace){
- int next_wp=ClosestWaypoint( x, y, gpsTrace);
- int prev_wp = next_wp-1;
- double n_x = gpsTrace[next_wp].x-gpsTrace[prev_wp].x;
- double n_y = gpsTrace[next_wp].y-gpsTrace[prev_wp].y;
- double x_x = x - gpsTrace[prev_wp].x;
- double x_y = y - gpsTrace[prev_wp].y;
-
- double proj_norm = (x_x*n_x+x_y*n_y)/(n_x*n_x+n_y*n_y);
- double proj_x = proj_norm*n_x;
- double proj_y = proj_norm*n_y;
- double frenet_d = sqrt((x_x - proj_x) * (x_x - proj_x) + (x_y - proj_y) * (x_y - proj_y));
-
-
-
- double center_x = 1000-gpsTrace[prev_wp].x;
- double center_y = 2000-gpsTrace[prev_wp].y;
- double centerToPos = distance(center_x,center_y,x_x,x_y);
- double centerToRef = distance(center_x,center_y,proj_x,proj_y);
- if(centerToPos <= centerToRef){
- frenet_d *= -1;
- }
-
- double frenet_s = 0;
- for(int i = 0; i < prev_wp; i++){
- frenet_s += distance(gpsTrace[i].x,gpsTrace[i].y,gpsTrace[i+1].x,gpsTrace[i+1].y);
- }
- if(prev_wp<=0){
- frenet_s -= distance(0,0,proj_x,proj_y);
- }else{
- frenet_s += distance(0,0,proj_x,proj_y);
- }
- double tmp_Angle = atan2(n_y,n_x);
- return {x:x,y:y,s:frenet_s,d:frenet_d,tangent_Ang:tmp_Angle};
- }
-
-
- void iv::decition::FrenetPlanner::Frenet2XY(double *res_x, double *res_y, double s, double d, const vector<iv::decition::FrenetPoint>& frenetTrace){
- int prev_wp = 0;
-
- while(s > frenetTrace[prev_wp+1].s && (prev_wp < (int)(frenetTrace.size()-2) )){
- prev_wp++;
- }
- int wp2 = prev_wp+1;
- double heading = atan2((frenetTrace[wp2].y-frenetTrace[prev_wp].y),(frenetTrace[wp2].x-frenetTrace[prev_wp].x));
- double seg_s = (s-frenetTrace[prev_wp].s);
- double seg_x = frenetTrace[prev_wp].x+seg_s*cos(heading);
- double seg_y = frenetTrace[prev_wp].y+seg_s*sin(heading);
- double perp_heading = heading-M_PI*0.5;
- *res_x = seg_x + d*cos(perp_heading);
- *res_y = seg_y + d*sin(perp_heading);
- }
- iv::decition::FrenetPoint iv::decition::FrenetPlanner::getFrenetfromXY(double x, double y, const std::vector<iv::Point2D>& gpsTrace,const std::vector<iv::GPSData>& gpsMap,int pathpoint,GPS_INS nowGps){
- int next_wp=ClosestWaypoint( x, y, gpsTrace);
- int prev_wp = next_wp-1;
-
-
-
- GPS_INS gps = Coordinate_UnTransfer(x,y,nowGps);
- Point2D pt = Coordinate_Transfer(gps.gps_x,gps.gps_y, *gpsMap[(pathpoint+prev_wp)%gpsMap.size()]);
-
- double frenet_s = 0;
- for(int i = 0; i < prev_wp; i++){
- frenet_s += distance(gpsTrace[i].x,gpsTrace[i].y,gpsTrace[i+1].x,gpsTrace[i+1].y);
- }
- frenet_s += pt.y;
-
- double theta = (nowGps.ins_heading_angle - gpsMap[(pathpoint+prev_wp)%gpsMap.size()]->ins_heading_angle+90);
- double tmp_Angle = theta * M_PI / 180;
- return {x:x,y:y,s:frenet_s,d:pt.x,tangent_Ang:tmp_Angle};
- }
- void iv::decition::FrenetPlanner::getXYfromFrenet(double *res_x, double *res_y, double s, double d, const vector<iv::decition::FrenetPoint>& frenetTrace,const std::vector<iv::GPSData>& gpsMap,int pathpoint,GPS_INS nowGps)
- {
- int prev_wp = -1;
- while((prev_wp < (int)(frenetTrace.size()-1) ) && s > frenetTrace[prev_wp+1].s)
- {
- prev_wp++;
- }
- if(prev_wp < 0){
- prev_wp = 0;
- }
- GPS_INS gps= Coordinate_UnTransfer( d, s-frenetTrace[prev_wp].s , *gpsMap[(pathpoint+prev_wp)%gpsMap.size()]);
- Point2D pt = Coordinate_Transfer( gps.gps_x, gps.gps_y, nowGps);
- *res_x = pt.x;
- *res_y = pt.y;
- }
- iv::decition::Quintic_polynomial::Quintic_polynomial(double xs, double vxs, double axs, double xe, double vxe, double axe, double T){
-
- this->xs = xs;
- this->vxs = vxs;
- this->axs = axs;
- this->xe = xe;
- this->vxe = vxe;
- this->axe = axe;
- this->a0 = xs;
- this->a1 = vxs;
- this->a2 = axs/ 2.0;
- MatrixXd A(3, 3);
- MatrixXd b(3, 1);
- MatrixXd x(3, 1);
- A << pow(T, 3), pow(T, 4), pow(T, 5),
- 3*pow(T, 2), 4*pow(T, 3), 5*pow(T, 4),
- 6*T, 12*T*T, 20*pow(T, 3);
- b << xe - a0 - a1*T - a2*T*T,
- vxe - a1 - 2*a2*T,
- axe - 2*a2;
- x=A.colPivHouseholderQr().solve(b);
- this->a3 = x(0,0);
- this->a4 = x(1,0);
- this->a5 = x(2,0);
- }
- iv::decition::Quintic_polynomial::~Quintic_polynomial(){
- }
- double iv::decition::Quintic_polynomial::calc_point(double t){
- double xt = this->a0 + this->a1 * t + this->a2 * t*t + this->a3 * t*t*t + this->a4 * t*t*t*t + this->a5 * t*t*t*t*t;
- return xt;
- }
- double iv::decition::Quintic_polynomial::calc_first_derivative(double t){
- double xt = this->a1 + 2*this->a2 * t + 3*this->a3 * t*t + 4*this->a4 * t*t*t + 5*this->a5 * t*t*t*t;
- return xt;
- }
- double iv::decition::Quintic_polynomial::calc_second_derivative(double t){
- double xt = 2*this->a2 + 6*this->a3 * t + 12*this->a4 * t*t + 20*this->a5 * t*t*t;
- return xt;
- }
- double iv::decition::Quintic_polynomial::calc_third_derivative(double t){
- double xt = 6*this->a3 + 24*this->a4 * t + 60*this->a5 * t*t;
- return xt;
- }
- iv::decition::Quartic_polynomial::Quartic_polynomial(double xs, double vxs, double axs, double vxe, double axe, double T){
-
- this->xs = xs;
- this->vxs = vxs;
- this->axs = axs;
- this->vxe = vxe;
- this->axe = axe;
- this->a0 = xs;
- this->a1 = vxs;
- this->a2 = axs / 2.0;
- MatrixXd A(2, 2);
- MatrixXd b(2, 1);
- MatrixXd x(2, 1);
- A << 3*pow(T, 2), 4*pow(T, 3),
- 6*T, 12*T*T ;
- b << vxe - a1 - 2*a2*T,
- axe - 2*a2;
- x=A.colPivHouseholderQr().solve(b);
- this->a3 = x(0,0);
- this->a4 = x(1,0);
- }
- iv::decition::Quartic_polynomial::~Quartic_polynomial(){
- }
- double iv::decition::Quartic_polynomial::calc_point(double t){
- double xt = this->a0 + this->a1*t + this->a2*t*t + this->a3*t*t*t + this->a4*t*t*t*t;
- return xt;
- }
- double iv::decition::Quartic_polynomial::calc_first_derivative(double t){
- double xt = this->a1 + 2*this->a2*t + 3*this->a3*t*t + 4*this->a4*t*t*t;
- return xt;
- }
- double iv::decition::Quartic_polynomial::calc_second_derivative(double t){
- double xt = 2*this->a2 + 6*this->a3*t + 12*this->a4*t*t;
- return xt;
- }
- double iv::decition::Quartic_polynomial::calc_third_derivative(double t){
- double xt = 6*this->a3 + 24*this->a4*t;
- return xt;
- }
- bool iv::decition::FrenetPlanner::comp(const iv::decition::Frenet_path &a, const iv::decition::Frenet_path &b){
- return a.cf < b.cf;
- }
|