You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

458 lines
15KB

  1. #include <stdint.h>
  2. #include <stdio.h>
  3. #include <stdlib.h>
  4. #include <string.h>
  5. #include <math.h>
  6. #include <sys/types.h>
  7. #include <fcntl.h>
  8. #include <sys/stat.h>
  9. #include <sys/time.h>
  10. #include <servo.h>
  11. #include <spi.h>
  12. #include <ik_double.h>
  13. #warning code uses old leg numbering!
  14. void move_to(struct bot_double*, struct bot_double*, int);
  15. void demo();
  16. int get_time();
  17. static struct bot_double startup_position = {
  18. {0,0,0}, // body offset
  19. {0,0,0}, // world position
  20. {0,0,0}, // body rotation
  21. {
  22. { // leg 0
  23. { 80, 40, 0}, // base offset
  24. {200,120,10}, // end effector position
  25. { 0, 0, 0} // leg angles
  26. },{ // leg 1
  27. { 0, 60, 0},
  28. { 0,200,10},
  29. { 0, 0, 0}, // leg angles
  30. },{
  31. {-80, 40, 0},
  32. {-200,120,10},
  33. { 0, 0, 0} // leg angles
  34. },{
  35. { 80,-40, 0},
  36. {200,-120,10},
  37. { 0, 0, 0} // leg angles
  38. },{
  39. { 0,-60, 0},
  40. {0,-200,10},
  41. { 0, 0, 0} // leg angles
  42. },{
  43. {-80,-40, 0},
  44. {-200,-120,10},
  45. { 0, 0, 0} // leg angles
  46. }}
  47. };
  48. static struct bot_double idle_position = {
  49. {0,0,0}, // body offset
  50. {0,0,0}, // world position
  51. {0,0,0}, // body rotation
  52. {
  53. { // leg 0
  54. { 80, 40, 0}, // base offset
  55. {150,120,120}, // end effector position
  56. { 0, 0, 0} // leg angles
  57. },{ // leg 1
  58. { 0, 60, 0},
  59. { 0,160,120},
  60. { 0, 0, 0}, // leg angles
  61. },{
  62. {-80, 40, 0}, // leg 2
  63. {-150,120,120},
  64. { 0, 0, 0} // leg angles
  65. },{
  66. { 80,-40, 0}, // leg 3
  67. {150,-120,120},
  68. { 0, 0, 0} // leg angles
  69. },{
  70. { 0,-60, 0},
  71. {0,-160,120},
  72. { 0, 0, 0} // leg angles
  73. },{
  74. {-80,-40, 0},
  75. {-150,-120,120},
  76. { 0, 0, 0} // leg angles
  77. }}
  78. };
  79. static struct bot_double current_position;
  80. static struct bot_double target_position;
  81. int
  82. main(int argc, char **argv) {
  83. int i,j;
  84. int frame;
  85. uint32_t time;
  86. printf("welcome.\n");
  87. spi_open(0,33000000);
  88. for(i=0;i<NUM_SERVOS;i++) servo[i]=0;
  89. spi_update_servos(servo);
  90. ik_double(&startup_position);
  91. for(i=0;i<NUM_LEGS;i++) {
  92. for(j=0;j<3;j++) {
  93. servo[i*3+j]=servo_directions[i*3+j]*(startup_position.leg[i].angle[j])*1800.0/M_PI+servo_offsets[i*3+j];
  94. }
  95. }
  96. spi_update_servos(servo);
  97. demo();
  98. spi_close();
  99. }
  100. void
  101. move_to(struct bot_double *current_position, struct bot_double *target_position, int num_frames) {
  102. struct bot_double frame_position;
  103. int frame;
  104. unsigned int frame_start, frame_end;
  105. struct timespec delay;
  106. int leg,i,j;
  107. memcpy(&frame_position,current_position,sizeof(struct bot_double));
  108. for(frame=1;frame<=num_frames;frame++) {
  109. frame_start=get_time();
  110. frame_position.position.x=current_position->position.x+frame*(target_position->position.x-current_position->position.x)/num_frames;
  111. frame_position.position.y=current_position->position.y+frame*(target_position->position.y-current_position->position.y)/num_frames;
  112. frame_position.position.z=current_position->position.z+frame*(target_position->position.z-current_position->position.z)/num_frames;
  113. frame_position.orientation.rx=current_position->orientation.rx+frame*(target_position->orientation.rx-current_position->orientation.rx)/num_frames;
  114. frame_position.orientation.ry=current_position->orientation.ry+frame*(target_position->orientation.ry-current_position->orientation.ry)/num_frames;
  115. frame_position.orientation.rz=current_position->orientation.rz+frame*(target_position->orientation.rz-current_position->orientation.rz)/num_frames;
  116. for(leg=0;leg<NUM_LEGS;leg++) {
  117. frame_position.leg[leg].position.x=current_position->leg[leg].position.x+frame*(target_position->leg[leg].position.x-current_position->leg[leg].position.x)/num_frames;
  118. frame_position.leg[leg].position.y=current_position->leg[leg].position.y+frame*(target_position->leg[leg].position.y-current_position->leg[leg].position.y)/num_frames;
  119. frame_position.leg[leg].position.z=current_position->leg[leg].position.z+frame*(target_position->leg[leg].position.z-current_position->leg[leg].position.z)/num_frames;
  120. }
  121. ik_double(&frame_position);
  122. for(i=0;i<NUM_LEGS;i++) {
  123. for(j=0;j<3;j++) {
  124. servo[i*3+j]=servo_directions[i*3+j]*(frame_position.leg[i].angle[j])*1800.0/M_PI+servo_offsets[i*3+j];
  125. }
  126. }
  127. frame_end=get_time();
  128. if(frame_end-frame_start>20000) {
  129. printf("frame %d delay %d\n",frame,frame_end-frame_start);
  130. }
  131. while(frame_end-frame_start<20000) {
  132. // delay.tv_sec=0;
  133. // delay.tv_nsec=(frame_end-frame_start)*1000;
  134. // nanosleep(&delay,NULL);
  135. frame_end=get_time();
  136. }
  137. spi_update_servos(servo);
  138. }
  139. memcpy(current_position,target_position,sizeof(struct bot_double));
  140. }
  141. void
  142. move_to_parabolic(struct bot_double *current_position, struct bot_double *target_position, int num_frames, double dx, double dy, double dz, double a, int leg_mask) {
  143. struct bot_double frame_position;
  144. int frame;
  145. int frame_start, frame_end;
  146. int leg,i,j;
  147. double t,d;
  148. memcpy(&frame_position,current_position,sizeof(struct bot_double));
  149. for(frame=1;frame<=num_frames;frame++) {
  150. frame_start=get_time();
  151. t=2.0*frame/num_frames-1.0;
  152. d=a-(a*t*t);
  153. frame_position.position.x=current_position->position.x+frame*(target_position->position.x-current_position->position.x)/num_frames;
  154. frame_position.position.y=current_position->position.y+frame*(target_position->position.y-current_position->position.y)/num_frames;
  155. frame_position.position.z=current_position->position.z+frame*(target_position->position.z-current_position->position.z)/num_frames;
  156. frame_position.orientation.rx=current_position->orientation.rx+frame*(target_position->orientation.rx-current_position->orientation.rx)/num_frames;
  157. frame_position.orientation.ry=current_position->orientation.ry+frame*(target_position->orientation.ry-current_position->orientation.ry)/num_frames;
  158. frame_position.orientation.rz=current_position->orientation.rz+frame*(target_position->orientation.rz-current_position->orientation.rz)/num_frames;
  159. for(leg=0;leg<NUM_LEGS;leg++) {
  160. if(!((1<<leg)&leg_mask)) {
  161. frame_position.leg[leg].position.x=current_position->leg[leg].position.x+frame*(target_position->leg[leg].position.x-current_position->leg[leg].position.x)/num_frames;
  162. frame_position.leg[leg].position.y=current_position->leg[leg].position.y+frame*(target_position->leg[leg].position.y-current_position->leg[leg].position.y)/num_frames;
  163. frame_position.leg[leg].position.z=current_position->leg[leg].position.z+frame*(target_position->leg[leg].position.z-current_position->leg[leg].position.z)/num_frames;
  164. } else {
  165. frame_position.leg[leg].position.x=dx*d + current_position->leg[leg].position.x + frame*(target_position->leg[leg].position.x - current_position->leg[leg].position.x)/num_frames;
  166. frame_position.leg[leg].position.y=dy*d + current_position->leg[leg].position.y + frame*(target_position->leg[leg].position.y - current_position->leg[leg].position.y)/num_frames;
  167. frame_position.leg[leg].position.z=dz*d + current_position->leg[leg].position.z + frame*(target_position->leg[leg].position.z - current_position->leg[leg].position.z)/num_frames;
  168. }
  169. }
  170. ik_double(&frame_position);
  171. for(i=0;i<NUM_LEGS;i++) {
  172. for(j=0;j<3;j++) {
  173. servo[i*3+j]=servo_directions[i*3+j]*(frame_position.leg[i].angle[j])*1800.0/M_PI+servo_offsets[i*3+j];
  174. }
  175. }
  176. frame_end=get_time();
  177. if(frame_end-frame_start>20000) {
  178. printf("frame %d delay %d\n",frame,frame_end-frame_start);
  179. }
  180. while(frame_end-frame_start<20000) {
  181. // delay.tv_sec=0;
  182. // delay.tv_nsec=(frame_end-frame_start)*1000;
  183. // nanosleep(&delay,NULL);
  184. frame_end=get_time();
  185. }
  186. spi_update_servos(servo);
  187. }
  188. memcpy(current_position,target_position,sizeof(struct bot_double));
  189. }
  190. void
  191. walk(double dx, double dy, double alpha, int frames, int mode) {
  192. int leg_mask;
  193. int i;
  194. double step_height=30;
  195. struct bot_double start_position;
  196. memcpy(&target_position, &current_position,sizeof(struct bot_double));
  197. memcpy(&start_position,&current_position,sizeof(struct bot_double));
  198. leg_mask=0;
  199. for(i=0;i<NUM_LEGS;i++) {
  200. if(i%2) {
  201. leg_mask|=(1<<i);
  202. target_position.leg[i].position.x=dx+(current_position.leg[i].position.x*cos(alpha/2)-current_position.leg[i].position.y*sin(alpha/2));
  203. target_position.leg[i].position.y=dy+(current_position.leg[i].position.x*sin(alpha/2)+current_position.leg[i].position.y*cos(alpha/2));
  204. }
  205. }
  206. move_to_parabolic(&current_position,&target_position,frames,0,0,-1,step_height,leg_mask);
  207. target_position.position.x=start_position.position.x+dx/2;
  208. target_position.position.y=start_position.position.y+dy/2;
  209. target_position.orientation.rz=start_position.orientation.rz+alpha/2;
  210. move_to(&current_position,&target_position,frames);
  211. leg_mask=0;
  212. for(i=0;i<NUM_LEGS;i++) {
  213. if(!(i%2)) {
  214. leg_mask|=(1<<i);
  215. target_position.leg[i].position.x=dx+(current_position.leg[i].position.x*cos(alpha/2)-current_position.leg[i].position.y*sin(alpha/2));
  216. target_position.leg[i].position.y=dy+(current_position.leg[i].position.x*sin(alpha/2)+current_position.leg[i].position.y*cos(alpha/2));
  217. }
  218. }
  219. move_to_parabolic(&current_position,&target_position,frames,0,0,-1,step_height,leg_mask);
  220. target_position.position.x=start_position.position.x+dx;
  221. target_position.position.y=start_position.position.y+dy;
  222. target_position.orientation.rz=start_position.orientation.rz+alpha;
  223. move_to(&current_position,&target_position,frames);
  224. memcpy(&target_position,&current_position,sizeof(struct bot_double));
  225. }
  226. void
  227. demo() {
  228. int i,j;
  229. // stand up
  230. memcpy(&current_position,&startup_position,sizeof(struct bot_double));
  231. memcpy(&target_position,&current_position,sizeof(struct bot_double));
  232. // lift up
  233. for(i=0;i<NUM_LEGS;i++) {
  234. target_position.leg[i].position.z=130;
  235. }
  236. move_to(&current_position,&target_position,50);
  237. // move legs to idle position
  238. for(i=0;i<(NUM_LEGS/2);i++) {
  239. memcpy(&target_position.leg[i],&idle_position.leg[i],sizeof(struct leg_double));
  240. memcpy(&target_position.leg[NUM_LEGS-1-i],&idle_position.leg[NUM_LEGS-1-i],sizeof(struct leg_double));
  241. move_to_parabolic(&current_position,&target_position,25,0,0,-1,30,(1<<i)|(1<<(NUM_LEGS-1-i)));
  242. }
  243. // walk a square
  244. move_to(&current_position,&target_position,25);
  245. walk(0,50,0,20,0);
  246. target_position.position.z=20;
  247. move_to(&current_position,&target_position,25);
  248. walk(-50,0,0,20,0);
  249. target_position.position.z=40;
  250. move_to(&current_position,&target_position,25);
  251. walk(0,-50,0,20,0);
  252. target_position.position.z=-20;
  253. move_to(&current_position,&target_position,50);
  254. walk(50,0,0,20,0);
  255. // downward spiral
  256. sleep(1);
  257. target_position.position.z=0;
  258. target_position.orientation.rx=5*M_PI/180.0;
  259. move_to(&current_position,&target_position,25);
  260. for(j=0;j<3;j++) {
  261. for(i=0;i<16;i++) {
  262. target_position.orientation.rx=sin(M_PI*i/8.0)*5.0*M_PI/180.0;
  263. target_position.orientation.ry=cos(M_PI*i/8.0)*5.0*M_PI/180.0;
  264. target_position.position.z=i+j*16;
  265. move_to(&current_position,&target_position,8);
  266. }
  267. }
  268. target_position.orientation.rx=0;
  269. target_position.orientation.ry=0;
  270. target_position.position.z=0;
  271. move_to(&current_position,&target_position,25);
  272. // shake it, baby
  273. for(i=0;i<4;i++) {
  274. target_position.position.z=10*i;
  275. if(i%2) {
  276. target_position.orientation.rz=5.0*M_PI/180.0;
  277. } else {
  278. target_position.orientation.rz=-5.0*M_PI/180;
  279. }
  280. move_to(&current_position,&target_position,25);
  281. }
  282. target_position.orientation.rz=0;
  283. move_to(&current_position,&target_position,5);
  284. sleep(1);
  285. // and some more..
  286. for(i=3;i>=0;i--) {
  287. target_position.position.z=10*i;
  288. if(i%2) {
  289. target_position.position.y=20;
  290. } else {
  291. target_position.position.y=-20;
  292. }
  293. move_to(&current_position,&target_position,25);
  294. }
  295. target_position.position.y=0;
  296. move_to(&current_position,&target_position,25);
  297. // lift middle legs
  298. target_position.leg[1].position.z-=80;
  299. target_position.leg[1].position.y+=20;
  300. target_position.leg[4].position.z-=80;
  301. target_position.leg[4].position.y-=20;
  302. move_to(&current_position,&target_position,100);
  303. target_position.leg[1].position.y+=80;
  304. target_position.leg[4].position.y-=80;
  305. move_to(&current_position,&target_position,100);
  306. target_position.position.z+=30;
  307. move_to(&current_position,&target_position,25);
  308. target_position.position.z-=60;
  309. move_to(&current_position,&target_position,50);
  310. target_position.position.z+=30;
  311. move_to(&current_position,&target_position,25);
  312. target_position.leg[1].position.y-=80;
  313. target_position.leg[4].position.y+=80;
  314. move_to(&current_position,&target_position,100);
  315. memcpy(&target_position,&idle_position,sizeof(struct bot_double));
  316. move_to(&current_position,&target_position,50);
  317. /*
  318. walk(50,50,-15*91,25,0);
  319. walk(50,50,-15*91,25,0);
  320. walk(-50,-50,15*91,25,0);
  321. walk(-50,-50,15*91,25,0);
  322. walk(50,-50,15*91,25,0);
  323. walk(50,-50,15*91,25,0);
  324. walk(-50,50,-15*91,25,0);
  325. walk(-50,50,-15*91,25,0);
  326. */
  327. // lean back and wave
  328. target_position.position.x=-40;
  329. move_to(&current_position,&target_position,25);
  330. sleep(1);
  331. target_position.leg[0].position.x=180;
  332. target_position.leg[0].position.y=150;
  333. target_position.leg[0].position.z=-120;
  334. target_position.leg[3].position.x=180;
  335. target_position.leg[3].position.y=-150;
  336. target_position.leg[3].position.z=-120;
  337. move_to(&current_position,&target_position,50);
  338. for(i=0;i<3;i++) {
  339. target_position.leg[0].position.x+=20;
  340. target_position.leg[0].position.y-=60;
  341. target_position.leg[3].position.x+=20;
  342. target_position.leg[3].position.y+=60;
  343. move_to(&current_position,&target_position,12);
  344. target_position.leg[0].position.y+=60;
  345. target_position.leg[0].position.x-=20;
  346. target_position.leg[3].position.y-=60;
  347. target_position.leg[3].position.x-=20;
  348. move_to(&current_position,&target_position,12);
  349. }
  350. sleep(1);
  351. memcpy(&target_position,&idle_position,sizeof(struct bot_double));
  352. target_position.position.x=-40;
  353. move_to(&current_position,&target_position,50);
  354. target_position.position.x=0;
  355. move_to(&current_position,&target_position,25);
  356. target_position.position.z=100;
  357. move_to(&current_position,&target_position,50);
  358. for(i=0;i<NUM_SERVOS;i++) { servo[i]=0; }
  359. spi_update_servos(servo);
  360. }
  361. int
  362. get_time() {
  363. struct timeval t;
  364. gettimeofday(&t,NULL);
  365. return (t.tv_sec*1000000+t.tv_usec);
  366. }