前言html
一.靜態障礙物的引入git
1.軌跡規劃框架
3、調試過程記錄性能
2.參數的調整優化
總結spa
前言
前情提要:Lattice Planner從學習到放棄
盲目激情的移植apollo的lattice,體會到了痛苦,當時爲了先儘快讓lattice算法在本身的框架裏跑起來,在摘用源碼的時候作了不少欠考慮的改動,雖然快速實現了經過lattice planner規劃軌跡,而後達到循跡的效果,可是在隨後引入障礙物,產生了大量的坑,深深的被教育,也給帶頭大哥添了很多麻煩。
.net
看懂流程==>理解原理 ==>成功實現
最終經過planner中的二次規劃實現了單車道內低速障礙物的避讓規劃。
簡明扼要:添加靜態障礙物→產生單車道內目標軌跡→繞行或中止
最近節奏比較撕裂,下午或晚上才能搞lattice的復現,但這個「重複造輪子」的過程真的獲益良多,和大哥們交流也多起來,知道好的輪子是什麼樣的,才能嘗試去改進或完善,一瓶不響半瓶咣噹,半瓶正是在下~保持積極與熱情,也期待大佬們各類指導....
一.靜態障礙物的引入
1.障礙物的格式
暫時不使用Apollo的那套,先用自有結構體接收障礙物信息,而後壓入Obstacle類中便可。
對於靜態或低速障礙物,無非就是id、長寬高、位置以及朝向,動態障礙物暫時沒有進行。
2.自車sl的創建和障礙物的創建
自車相對於參考線起點裏程和橫向偏移量,以及障礙物的ST、SL信息都是要計算的,對於動態障礙物,SL已經不足以知足需求了吧?後續學習EM再拎出來分析。
二.單車道場景中面向靜態障礙物的避讓
1.軌跡規劃原理
軌跡=縱向軌跡+橫向軌跡,以前已經分析了縱向速度規劃的過程,以及橫縱向軌跡是如何combine成一條完整軌跡的。在lattice中,橫向軌跡生成有兩種方法:撒點採樣法、二次規劃法。
1.1基於採樣點的軌跡規劃
撒點法前邊也有記錄,知道了起始點和採樣點的,而後求解對應的五次多項式的係數,即可獲得對應的橫向軌跡,不重複了。
1.2基於二次規劃的軌跡規劃
Apollo源碼中FLAGS_lateral_optimization默認是開啓的,即橫向規劃默認使用二次規劃進行,因爲我的魯莽認爲撒點會更直觀簡潔,因此關了這個標誌位,採用的是撒點採樣,隨後在實車調試時發現耗時很大,尤爲是自車接近障礙物時,耗時飆升——軌跡數太多,障礙物碰撞檢測遍歷軌跡。額,配置比較低的工控機雪上加霜...而後試用一下二次規劃唄,wtf,超nice。
學習了程十三的軌跡規劃綜述。
Apollo中二次規劃求解使用的是OSQP求解器,官網:OSQP官網點這裏 ,二次規劃求解有不少方法,關於爲什麼選取OSQP,速度快~爲什麼這麼快,數學系的大佬們的戰場。標準的二次規劃形式:
下來要作的就很明確了:
- 搞清楚路徑規劃的約束;
- 搞清楚優化目標,即如何評價計算結果的優劣
- 轉換成osqp求解器須要的形式,調用求解器
- 坐等結果~
考慮車寬以及和障礙物的距離buffer,車輛的中心的橫向可移動範圍大概就是以下圖,Apollo中,前探60m,採樣點間隔1m,因此有60個採樣點,也能夠根據實際狀況調整。
1.2.1 車輛橫向狀態量設計
橫向偏移量做爲基本量,跑不掉,做爲橫向速度變化率,固然也跑不掉;做爲橫向加加速度,是控制乘坐溫馨性的重要指標,也不能放掉;對於,把軌跡看做可拉伸的彈力繩,三階導意味着其可拉伸的程度,能夠用差分計算。so,須要的量齊全了。狀態量能夠獲得:
1.2.2 車輛橫向軌跡約束的創建
整個約束創建主要考慮到:
1.自車不能與障礙物碰撞或駛出邊界,即
2.軌跡應該保持連續,即相鄰兩個採樣點的
3.軌跡應該保持光滑,即導數連續相鄰兩個採樣點的斜率
4.橫向加加速度(相對於s的二階偏導)應在必定範圍內,即
公式二、3直接用的泰勒公式進行的2階和3階展開,將代入,可獲得公式2和3的最終形式:
到這一步,基本上約束的內容已經很清晰了,整理後以下:
即對於每個採樣點,存在兩個不等式和兩個等式約束,一共60個採樣點,那麼約束至少應該爲60x4=240個約束條件。
根據狀態量的形式,約束矩陣也就定了(這裏不必定準確,如有不對求指導額..):
這裏使用的話貌似應該是,行列貌似反了...無論了,原理不變。
1.2.3 約束邊界的創建
前邊已經描述了約束的創建依據,根據障礙物對車道的侵佔狀況,更新橫向位移的範圍,最終約束的上下邊界分別爲:
其中-2,2做爲缺省值不充值,湊夠矩陣運算數量,最終用於:
1.2.3 目標函數的創建
二次規劃的目標爲
其中,將權重做爲P項對角陣傳入,左右邊界做爲誤差項q用以控制軌跡和參考線的偏離程度,以下:
至此,整個二次規劃創建所須要的材料,所有齊全。在二次規劃中,對於矩陣P,以目前形式來看必定是正定的額,是否意味着此問題是凸優化問題,且必定有可行解?求大佬指點。。。
1.2.5 約束矩陣的壓縮CSC
從上邊已經能夠看到,創建起來的矩陣基本都很是稀疏,在運算過程當中是很是不方便的,常見的稀疏矩陣壓縮方法主要有:
- CSR—Compressed sparse row
- CSC—Compressed sparse column
以前在學習apollo時,發現其採用的csc矩陣,當時並不瞭解,也是擴展後才知道,可見總結:csc_matrix稀疏矩陣理解 至於爲什麼選擇csc的緣由多是osqp官方使用的是csc?調用osqp創建workspace,須要以csc矩陣形式傳入求解器。
閒話一大堆,下面上代碼。
2.Lattice planning二次規劃代碼實現
lattice planner中經過調用Trajectory1dGenerator實例化後的成員函數,生成橫縱向軌跡,咱們要看的常規撒點法和二次規劃都在其中,
// 5. generate 1d trajectory bundle for longitudinal and lateral respectively. Trajectory1dGenerator trajectory1d_generator( init_s, init_d, ptr_path_time_graph, ptr_prediction_querier); std::vector<std::shared_ptr<Curve1d>> lon_trajectory1d_bundle; std::vector<std::shared_ptr<Curve1d>> lat_trajectory1d_bundle; trajectory1d_generator.GenerateTrajectoryBundles( planning_target, &lon_trajectory1d_bundle, &lat_trajectory1d_bundle);
進入函數後,很清晰沒啥說的:縱向規劃+橫向規劃。
void Trajectory1dGenerator::GenerateTrajectoryBundles( const PlanningTarget& planning_target, Trajectory1DBundle* ptr_lon_trajectory_bundle, Trajectory1DBundle* ptr_lat_trajectory_bundle) { //縱向速度規劃 GenerateLongitudinalTrajectoryBundle(planning_target, ptr_lon_trajectory_bundle); //橫向軌跡規劃 GenerateLateralTrajectoryBundle(ptr_lat_trajectory_bundle); }
在橫向規劃內,經過宏定義FLAGS_lateral_optimization,決定是否採用二次規劃,以下:
void Trajectory1dGenerator::GenerateLateralTrajectoryBundle( Trajectory1DBundle* ptr_lat_trajectory_bundle) const { //是否使用優化軌跡,true,採用五次多項式規劃 if (!FLAGS_lateral_optimization) { auto end_conditions = end_condition_sampler_.SampleLatEndConditions(); // Use the common function to generate trajectory bundles. GenerateTrajectory1DBundle<5>(init_lat_state_, end_conditions, ptr_lat_trajectory_bundle); } else { double s_min = init_lon_state_[0]; double s_max = s_min + FLAGS_max_s_lateral_optimization;//FLAGS_max_s_lateral_optimization = 60 double delta_s = FLAGS_default_delta_s_lateral_optimization;//規劃間隔爲1m //橫向邊界 auto lateral_bounds = ptr_path_time_graph_->GetLateralBounds(s_min, s_max, delta_s); // LateralTrajectoryOptimizer lateral_optimizer; std::unique_ptr<LateralQPOptimizer> lateral_optimizer( new LateralOSQPOptimizer); // 採用的是OSQP求解器 lateral_optimizer->optimize(init_lat_state_, delta_s, lateral_bounds); auto lateral_trajectory = lateral_optimizer->GetOptimalTrajectory(); ptr_lat_trajectory_bundle->push_back( std::make_shared<PiecewiseJerkTrajectory1d>(lateral_trajectory)); } }
流程很直白,經過GetLateralBounds函數獲取包含障礙物信息的橫向邊界分佈,而後傳入lateral_optimizer->optimize()中開始osqp短暫愉快的一輩子,
完整源碼可看apollo:
bool LateralOSQPOptimizer::optimize( const std::array<double, 3>& d_state, const double delta_s, const std::vector<std::pair<double, double>>& d_bounds) { std::vector<c_float> P_data; std::vector<c_int> P_indices; std::vector<c_int> P_indptr; //創建目標函數中的P矩陣,主要包括權重分配 CalculateKernel(d_bounds, &P_data, &P_indices, &P_indptr); delta_s_ = delta_s; //1m const int num_var = static_cast<int>(d_bounds.size()); const int kNumParam = 3 * static_cast<int>(d_bounds.size()); const int kNumConstraint = kNumParam + 3 * (num_var - 1) + 3; c_float lower_bounds[kNumConstraint]; c_float upper_bounds[kNumConstraint]; const int prime_offset = num_var; const int pprime_offset = 2 * num_var;//=6? std::vector<std::vector<std::pair<c_int, c_float>>> columns; columns.resize(kNumParam); int constraint_index = 0; //constraint_index:0~2 // d_i+1'' - d_i'' for (int i = 0; i + 1 < num_var; ++i) { columns[pprime_offset + i].emplace_back(constraint_index, -1.0); columns[pprime_offset + i + 1].emplace_back(constraint_index, 1.0); //FLAGS_lateral_third_order_derivative_max=0.1 lower_bounds[constraint_index] = -FLAGS_lateral_third_order_derivative_max * delta_s_; upper_bounds[constraint_index] = FLAGS_lateral_third_order_derivative_max * delta_s_; ++constraint_index; } //constraint_index:3~5 // d_i+1' - d_i' - 0.5 * ds * (d_i'' + d_i+1'') for (int i = 0; i + 1 < num_var; ++i) { columns[prime_offset + i].emplace_back(constraint_index, -1.0); columns[prime_offset + i + 1].emplace_back(constraint_index, 1.0); columns[pprime_offset + i].emplace_back(constraint_index, -0.5 * delta_s_); columns[pprime_offset + i + 1].emplace_back(constraint_index, -0.5 * delta_s_); lower_bounds[constraint_index] = 0.0; upper_bounds[constraint_index] = 0.0; ++constraint_index; } //constraint_index:6~8 // d_i+1 - d_i - d_i' * ds - 1/3 * d_i'' * ds^2 - 1/6 * d_i+1'' * ds^2 for (int i = 0; i + 1 < num_var; ++i) { columns[i].emplace_back(constraint_index, -1.0); columns[i + 1].emplace_back(constraint_index, 1.0); columns[prime_offset + i].emplace_back(constraint_index, -delta_s_); columns[pprime_offset + i].emplace_back(constraint_index, -delta_s_ * delta_s_ / 3.0); columns[pprime_offset + i + 1].emplace_back(constraint_index, -delta_s_ * delta_s_ / 6.0); lower_bounds[constraint_index] = 0.0; upper_bounds[constraint_index] = 0.0; ++constraint_index; } columns[0].emplace_back(constraint_index, 1.0); lower_bounds[constraint_index] = d_state[0];//d upper_bounds[constraint_index] = d_state[0]; ++constraint_index; columns[prime_offset].emplace_back(constraint_index, 1.0); lower_bounds[constraint_index] = d_state[1];//d' upper_bounds[constraint_index] = d_state[1]; ++constraint_index; columns[pprime_offset].emplace_back(constraint_index, 1.0); lower_bounds[constraint_index] = d_state[2];//d'' upper_bounds[constraint_index] = d_state[2]; ++constraint_index; const double LARGE_VALUE = 2.0; for (int i = 0; i < kNumParam; ++i) { columns[i].emplace_back(constraint_index, 1.0); if (i < num_var) { lower_bounds[constraint_index] = d_bounds[i].first; upper_bounds[constraint_index] = d_bounds[i].second; } else { lower_bounds[constraint_index] = -LARGE_VALUE; upper_bounds[constraint_index] = LARGE_VALUE; } ++constraint_index; } CHECK_EQ(constraint_index, kNumConstraint); // change affine_constraint to CSC format std::vector<c_float> A_data; std::vector<c_int> A_indices; std::vector<c_int> A_indptr; int ind_p = 0; for (int j = 0; j < kNumParam; ++j) { A_indptr.push_back(ind_p); for (const auto& row_data_pair : columns[j]) { A_data.push_back(row_data_pair.second); A_indices.push_back(row_data_pair.first); ++ind_p; } } A_indptr.push_back(ind_p); // offset double q[kNumParam]; for (int i = 0; i < kNumParam; ++i) { if (i < num_var) { q[i] = -2.0 * FLAGS_weight_lateral_obstacle_distance * (d_bounds[i].first + d_bounds[i].second); } else { q[i] = 0.0; } } // Problem settings OSQPSettings* settings = reinterpret_cast<OSQPSettings*>(c_malloc(sizeof(OSQPSettings))); // Define Solver settings as default osqp_set_default_settings(settings); settings->alpha = 1.0; // Change alpha parameter settings->eps_abs = 1.0e-05; settings->eps_rel = 1.0e-05; settings->max_iter = 5000; settings->polish = true; settings->verbose = FLAGS_enable_osqp_debug; // Populate data OSQPData* data = reinterpret_cast<OSQPData*>(c_malloc(sizeof(OSQPData))); data->n = kNumParam; data->m = kNumConstraint; data->P = csc_matrix(data->n, data->n, P_data.size(), P_data.data(), P_indices.data(), P_indptr.data()); data->q = q; data->A = csc_matrix(data->m, data->n, A_data.size(), A_data.data(), A_indices.data(), A_indptr.data()); data->l = lower_bounds; data->u = upper_bounds; // Workspace OSQPWorkspace* work = osqp_setup(data, settings); // Solve Problem osqp_solve(work); // extract primal results // prime求導符號 for (int i = 0; i < num_var; ++i) { opt_d_.push_back(work->solution->x[i]); opt_d_prime_.push_back(work->solution->x[i + num_var]); opt_d_pprime_.push_back(work->solution->x[i + 2 * num_var]); } opt_d_prime_[num_var - 1] = 0.0; opt_d_pprime_[num_var - 1] = 0.0; // Cleanup osqp_cleanup(work); c_free(data->A); c_free(data->P); c_free(data); c_free(settings); return true; }
3、調試過程記錄
1.代碼缺失與完善
調試過程當中發現各類遺漏或者錯誤,改就是了...
2.實車執行失敗案例與分析
記錄下調試過程當中失敗的過程。
2.1 車輛遇到貼近的障礙物:一直想往上撞
計算耗時沒法控制,一旦接近障礙物,因爲備選軌跡不少,而且每一個軌跡都要進行碰撞檢測,外加工控機性能有限,致使耗時飆至900ms,已經嚴重不符100ms的運算週期了,直接致使軌跡拼接不許確,控制和規劃沒法很好銜接。
可是有一點沒想通:即便軌跡拼接出問題,爲什麼會往障礙物上撞,暫時把這個問題擱置了,後續解決。
2.2 使用二次規劃,貼近障礙物時無解,軌跡飛掉
(a)權重不合理,致使軌跡規劃無解
發如今和障礙物平齊時,自車橫向偏移量老是比邊界多了0.1m....而後求解失敗
畫個圖,一切明瞭,就像那高爾夫球進洞,或者《信條》裏的逆向子彈,若是軌跡有誤差,子彈是沒法退回槍膛的,只會和槍管發生碰撞。
解決方法比較簡單,調節權重,提升自車和障礙物距離的權重,使得生成的軌跡適當遠離障礙物,從紅色變爲綠色線。
(b)過早的轉彎,致使與障礙物發生碰撞
這裏問題根源還沒鎖定,可能因素有兩塊:
- 控制在選取預瞄距離後,過早的進行了轉向控制
- 規劃的軌跡的確過早轉向(這樣的話就不知足求解約束了額,可能性不大)
不知有前輩怎麼解決的。
2.3 在特定的幾個點,軌跡直接飛掉,約束失效
車輛到某些位置會出現規劃失敗的情形,沒法產生有效軌跡。爲了查緣由,關閉了軌跡有效性檢測,而後發現軌跡是這個diao樣子....飛掉了
查看該處的障礙物SL信息,發現SL徹底亂掉了。回想Frenet座標系在面對U型彎圓心處障礙物投影特色(障礙物會被極大拉伸),在出問題的點,該位置處參考線爲圓弧形,障礙物剛好位於其圓心附近,致使障礙物在frennet投影時出現了右下圖所示,存在多個投影點,其SL圖固然是要廢掉了。
總結
目前已經實現了單車道內障礙物的規避,包括繞行和減速以及停車。一套下來,雖然只是簡單的功能復現,僅僅冰山一角但已經收益頗豐,實際操做的過程難度和踩坑大幅超過了本身的預期,多虧了身邊大哥的指導和幫助,開發工做真的是須要多交流與溝通。
下一階段目標:
實現多車道的軌跡規劃,園區內能夠採用僞車道,即單條referenceline進行擴幅,覆蓋整條道路寬度便可,另外一個是真實的多車道多referenceline,繼續加油,一點點進步。