00001
00023 #include "olfati_saber_flocking/olfati_saber_flocking.h"
00024
00025 #define PI 3.14159265358979323846
00026 #define EPSILON 0.1
00027 #define A 5
00028 #define B 5
00029 const double C = abs(A-B) / sqrt(4*A*B);
00030 #define H 0.2
00031 #define D 10
00032 #define R 12
00033 #define C1 0.3
00034 #define C2 0.3
00035
00036 double pm1=1,pm2=3,pm3=1;
00037
00038 using namespace std;
00039
00040 class NeighborHandle
00041 {
00042 public:
00043 double _px,_py,_vx,_vy;
00044 pair<double,double> _position,_velocity;
00045 int _r_id;
00046 double mypm_g;
00047 NeighborHandle(int r_id, float x, float y, float vx, float vy)
00048 {
00049
00050 _px=x;
00051 _py=y;
00052 _vx=vx;
00053 _vy=vy;
00054 _position=pair<double,double>(x,y);
00055 _velocity=pair<double,double>(vx,vy);
00056 _r_id = r_id;
00057 mypm_g=1;
00058 }
00059 };
00060
00061 static list<NeighborHandle*> neighbor_list;
00062 pair<double,double> my_position=pair<double,double>(0,0);
00063 pair<double,double> my_velocity=pair<double,double>(0,0);
00064
00065 pair<double,double> get_vector(pair<double,double> start,pair<double,double> end)
00066 {
00067 pair<double,double> re=pair<double,double>(0,0);
00068 re.first=end.first-start.first;
00069 re.second=end.second-start.second;
00070 return re;
00071 }
00072
00073 double segma_norm(pair<double,double> v)
00074 {
00075 double re = EPSILON*(v.first*v.first+v.second*v.second);
00076 re = sqrt(1+re)-1;
00077 re /= EPSILON;
00078 return re;
00079 }
00080
00081 double R_alpha = segma_norm(pair<double,double>(R,0));
00082 double D_alpha = segma_norm(pair<double,double>(D,0));
00083
00084 pair<double,double> segma_epsilon(pair<double,double> v)
00085 {
00086 pair<double,double> re = pair<double,double>(0,0);
00087 double scale = 1+EPSILON*(v.first*v.first+v.second*v.second);
00088 scale = sqrt(scale);
00089 re.first = v.first / scale;
00090 re.second = v.second / scale;
00091 return re;
00092 }
00093
00094 double segma_1(double z)
00095 {
00096 return z / sqrt(1+z*z);
00097 }
00098
00099 double phi(double z)
00100 {
00101 return 0.5*((A+B)*segma_1(z+C)+A-B);
00102 }
00103
00104 double rho(double z)
00105 {
00106 if(z<H)
00107 return 1;
00108 if(z>1)
00109 return 0;
00110 return 0.5*(1+cos(PI*(z-H)/(1-H)));
00111 }
00112
00113 double phi_alpha(double z)
00114 {
00115 return rho(z/R_alpha)*phi(z-D_alpha);
00116 }
00117
00118 pair<double,double> f_g()
00119 {
00120 pair<double,double> re = pair<double,double>(0,0);
00121 for(list<NeighborHandle*>::iterator i=neighbor_list.begin();i!=neighbor_list.end();i++)
00122 {
00123 pair<double,double> q_ij = get_vector(my_position,(*i)->_position);
00124 pair<double,double> n_ij = segma_epsilon(q_ij);
00125 re.first += phi_alpha(segma_norm(q_ij))*n_ij.first*(*i)->mypm_g;
00126 re.second += phi_alpha(segma_norm(q_ij))*n_ij.second*(*i)->mypm_g;
00127 }
00128 return re;
00129 }
00130
00131 double a_ij(pair<double,double> j_p)
00132 {
00133 return rho(segma_norm(get_vector(my_position,j_p)) / R_alpha);
00134 }
00135
00136 pair<double,double> f_d()
00137 {
00138 pair<double,double> re = pair<double,double>(0,0);
00139 for(list<NeighborHandle*>::iterator i=neighbor_list.begin();i!=neighbor_list.end();i++)
00140 {
00141 pair<double,double> p_ij = get_vector(my_velocity,(*i)->_velocity);
00142 re.first += a_ij((*i)->_position) * p_ij.first;
00143 re.second += a_ij((*i)->_position) * p_ij.second;
00144 }
00145 return re;
00146 }
00147
00148 double interval = 0.01;
00149 pair<double,double> f_r()
00150 {
00151 pair<double,double> re = pair<double,double>(0,0);
00152 static pair<double,double> q_r = pair<double,double>(0,0);
00153 pair<double,double> p_r = pair<double,double>(0.5,0.5);
00154 q_r.first += p_r.first * interval;
00155 q_r.second += p_r.second * interval;
00156 re.first = -C1*get_vector(q_r,my_position).first - C2*get_vector(p_r,my_velocity).first;
00157 re.second = -C1*get_vector(q_r,my_position).second - C2*get_vector(p_r,my_velocity).second;
00158 return re;
00159 }
00160
00161
00162 PLUGINLIB_EXPORT_CLASS(olfati_saber_flocking::OlfatiSaberFlocking, micros_swarm::Application)
00163
00164 namespace olfati_saber_flocking{
00165
00166 OlfatiSaberFlocking::OlfatiSaberFlocking() {}
00167
00168 OlfatiSaberFlocking::~OlfatiSaberFlocking() {}
00169
00170 void OlfatiSaberFlocking::stop() {}
00171
00172 void OlfatiSaberFlocking::init()
00173 {
00174
00175 hz=10;
00176 interval=1.0/hz;
00177 }
00178
00179 void OlfatiSaberFlocking::publish_cmd(const ros::TimerEvent&)
00180 {
00181 geometry_msgs:: Twist msg;
00182 micros_swarm::Neighbors<micros_swarm::NeighborBase> n(true);
00183
00184 typename std::map<int, micros_swarm::NeighborBase>::iterator it;
00185 for(it=n.data().begin();it!=n.data().end();it++)
00186 {
00187 NeighborHandle* nh=new NeighborHandle(it->first, it->second.x, it->second.y, it->second.vx, it->second.vy);
00188 neighbor_list.push_back(nh);
00189 }
00190
00191 micros_swarm::Base nl = get_base();
00192
00193 my_position=pair<double,double>(nl.x, nl.y);
00194 my_velocity=pair<double,double>(nl.vx, nl.vy);
00195
00196 msg.linear.x += (f_g().first*pm1+f_d().first*pm2+f_r().first*pm3)/hz;
00197 msg.linear.y += (f_g().second*pm1+f_d().second*pm2+f_r().second*pm3)/hz;
00198
00199
00200 if (msg.linear.x >1)
00201 msg.linear.x=1;
00202 if (msg.linear.x <-1)
00203 msg.linear.x=-1;
00204 if (msg.linear.y >1)
00205 msg.linear.y=1;
00206 if (msg.linear.y <-1)
00207 msg.linear.y=-1;
00208 pub.publish(msg);
00209
00210 neighbor_list.clear();
00211 }
00212
00213 void OlfatiSaberFlocking::baseCallback(const nav_msgs::Odometry& lmsg)
00214 {
00215 float x=lmsg.pose.pose.position.x;
00216 float y=lmsg.pose.pose.position.y;
00217
00218 float vx=lmsg.twist.twist.linear.x;
00219 float vy=lmsg.twist.twist.linear.y;
00220
00221 micros_swarm::Base l(x, y, 0, vx, vy, 0, 1);
00222 set_base(l);
00223 }
00224
00225 void OlfatiSaberFlocking::start()
00226 {
00227 init();
00228
00229 ros::NodeHandle nh;
00230 set_dis(12);
00231 pub = nh.advertise<geometry_msgs::Twist>("cmd_vel", 1000);
00232 sub = nh.subscribe("base_pose_ground_truth", 1000, &OlfatiSaberFlocking::baseCallback, this, ros::TransportHints().udp());
00233 ros::Duration(5).sleep();
00234
00235 timer = nh.createTimer(ros::Duration(interval), &OlfatiSaberFlocking::publish_cmd, this);
00236 }
00237 };
00238