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.

233 lines
6.2KB

  1. #include <unistd.h>
  2. #include <math.h>
  3. #include <stdio.h>
  4. #include <fcntl.h>
  5. #include <termios.h>
  6. #include <string.h>
  7. #include <stdlib.h>
  8. #include <sched.h>
  9. #include <sys/time.h>
  10. #include <sys/types.h>
  11. #include <sys/socket.h>
  12. #include <sys/poll.h>
  13. #include <sys/un.h>
  14. #include <sys/resource.h>
  15. #include "servo.h"
  16. #include "spi.h"
  17. #include "ik.h"
  18. #include "dynamic_sequencer.h"
  19. #include "licks_message.h"
  20. struct sched_param sched_p;
  21. struct timespec timeout;
  22. static licks_message message;
  23. void ik_to_servos(bot *);
  24. long get_time();
  25. static bot bot_position = {
  26. {0,0,0}, // world position
  27. {0,0,0}, // world orientation
  28. {0,0,-20}, // body position
  29. {0,0,0}, // body orientation
  30. { // leg positions
  31. {{ 166, 110, 0},}, // leg 0
  32. {{ 0, 160, 0},}, // leg 1
  33. {{-166, 110, 0},}, // ...
  34. {{-166,-110, 0},},
  35. {{ 0,-160, 0},},
  36. {{ 166,-110, 0},}
  37. }
  38. };
  39. int
  40. main(int argc, char **argv) {
  41. sequencer_walk_parameters wp;
  42. vector3d move_vector, rotate_vector;
  43. char *endptr;
  44. int i,l,d;
  45. int power=0;
  46. int quit=0;
  47. int frame=0;
  48. int query=0;
  49. long t_frame_start, t_frame;
  50. char *sender;
  51. licks_socket_open();
  52. load_calibration("/etc/calibration.bin");
  53. spi_open(0,15000000);
  54. #ifdef LINUX
  55. setpriority(PRIO_PROCESS,0,-99);
  56. sched_p.sched_priority=99;
  57. if(sched_setscheduler(0,SCHED_FIFO,&sched_p)==-1) {
  58. perror("sched_setscheduler");
  59. }
  60. #endif
  61. sequencer_init();
  62. wp.step_direction.x=0;
  63. wp.step_direction.y=0;
  64. wp.step_direction.z=0;
  65. wp.step_rotation=0;
  66. wp.step_duration=25;
  67. wp.step_height=20;
  68. sequencer_walk(&wp);
  69. while(!quit) {
  70. t_frame_start=get_time();
  71. sequencer_run_frame(&bot_position);
  72. ik(&bot_position);
  73. if(power) {
  74. ik_to_servos(&bot_position);
  75. } else {
  76. for(i=0;i<NUM_SERVOS;i++) servo_pwm[i]=0;
  77. spi_update_servos();
  78. }
  79. t_frame=get_time()-t_frame_start;
  80. while(licks_socket_poll()>0) {
  81. sender=receive_message(&message);
  82. switch(message.type) {
  83. case MSG_HELO:
  84. printf("HELO received\n");
  85. message.type='A';
  86. sprintf(message.parameters,"AH2U2");
  87. send_reply(&message);
  88. break;
  89. case MSG_QUIT:
  90. quit=1;
  91. break;
  92. case MSG_POWER:
  93. if(message.parameters[0]=='1') power=1; else power=0;
  94. break;
  95. case MSG_QUERY:
  96. printf("query\n");
  97. if(message.parameters[0]=='1') query=1; else query=0;
  98. break;
  99. case MSG_BODY:
  100. d=strtol(&(message.parameters[0]),&endptr,0);
  101. move_vector.x=strtod(endptr+1,&endptr);
  102. move_vector.y=strtod(endptr+1,&endptr);
  103. move_vector.z=strtod(endptr+1,&endptr);
  104. rotate_vector.x=strtod(endptr+1,&endptr);
  105. rotate_vector.y=strtod(endptr+1,&endptr);
  106. rotate_vector.z=strtod(endptr+1,&endptr);
  107. sequencer_move_body(0,d,&move_vector);
  108. sequencer_rotate_body(0,d,&rotate_vector);
  109. break;
  110. case MSG_LEG:
  111. d=strtol(&(message.parameters[0]),&endptr,0);
  112. l=strtol(endptr+1,&endptr,0);
  113. move_vector.x=strtod(endptr+1,&endptr);
  114. move_vector.y=strtod(endptr+1,&endptr);
  115. move_vector.z=strtod(endptr+1,&endptr);
  116. sequencer_move_leg(0,d,l,&move_vector);
  117. break;
  118. case MSG_GAIT:
  119. printf("gait\n");
  120. wp.step_duration=strtol(message.parameters,&endptr,0);
  121. wp.step_direction.x=strtod(endptr+1,&endptr);
  122. wp.step_direction.y=strtod(endptr+1,&endptr);
  123. wp.step_direction.z=strtod(endptr+1,&endptr);
  124. wp.step_rotation=strtod(endptr+1,&endptr);
  125. wp.step_height=strtol(endptr+1,&endptr,0);
  126. break;
  127. }
  128. }
  129. if(query&&((frame%10)==0)) {
  130. message.type='A';
  131. snprintf(message.parameters,255,
  132. "B%4.2f %4.2f %4.2f %2.3f %2.3f %2.3f",
  133. bot_position.body_position.x,
  134. bot_position.body_position.y,
  135. bot_position.body_position.z,
  136. bot_position.body_orientation.x,
  137. bot_position.body_orientation.y,
  138. bot_position.body_orientation.z);
  139. send_reply(&message);
  140. for(i=0;i<NUM_LEGS;i++) {
  141. snprintf(message.parameters,255,
  142. "L%d %4.2f %4.2f %4.2f %2.4f %2.4f %2.4f", i,
  143. bot_position.leg[i].position.x,
  144. bot_position.leg[i].position.y,
  145. bot_position.leg[i].position.z,
  146. bot_position.leg[i].ik_angle[0],
  147. bot_position.leg[i].ik_angle[1],
  148. bot_position.leg[i].ik_angle[2]
  149. );
  150. send_reply(&message);
  151. }
  152. snprintf(message.parameters,255,
  153. "G%d %4.2f %4.2f %4.2f %2.4f %4.2f",
  154. wp.step_duration,
  155. wp.step_direction.x,
  156. wp.step_direction.y,
  157. wp.step_direction.z,
  158. wp.step_rotation,
  159. wp.step_height
  160. );
  161. send_reply(&message);
  162. }
  163. t_frame=get_time()-t_frame_start;
  164. timeout.tv_sec=0;
  165. timeout.tv_nsec=1000*(14000-t_frame);
  166. nanosleep(&timeout,NULL);
  167. while((t_frame=get_time()-t_frame_start)<20000);
  168. // if(t_frame>21000) printf("slack: %d\n",t_frame);
  169. frame++;
  170. }
  171. spi_close();
  172. licks_socket_close();
  173. return 0;
  174. }
  175. void
  176. ik_to_servos(bot *b) {
  177. int i,j;
  178. for(i=0;i<NUM_LEGS;i++) {
  179. for(j=0;j<3;j++) {
  180. if(!isnan(b->leg[i].ik_angle[j])) {
  181. servo_pwm[i*3+j]=b->leg[i].ik_angle[j]*8.5*1800.0/M_PI+servo_offsets[i*3+j];
  182. }
  183. }
  184. }
  185. spi_update_servos();
  186. }
  187. long
  188. get_time() {
  189. struct timeval t;
  190. gettimeofday(&t,NULL);
  191. return (t.tv_sec*1000000+t.tv_usec);
  192. }