servocom_app.c 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440
  1. #include "global.h"
  2. #include "servocom_app.h"
  3. servo_param_t servo_x,servo_y,servo_z;
  4. //伺服通讯命令反馈。
  5. char servo_com_cmd_fb[SERVO_COM_CMD_NUMBER+1];
  6. //华创伺服参数读取列表
  7. servo_com_cmd_t servo_com_cmd_table[SERVO_AXISNUM][SERVO_COM_CMD_NUMBER] ={{
  8. //命令编码-------目标地址-------------通讯个数--------------数值地址------------------ID--------------CMD-cycle-time---回调函数
  9. {HANDSHAKE, SERVO_ADDR_HAND, 1,(unsigned char *)&servo_x.handshake, SERVO_X_COM_ID,0x03,0,50,servo_com_respone},
  10. {DI_MODE_SW, SERVO_ADDR_DI_MODE_SW, 2,(unsigned char *)&servo_x.di_function_set, SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  11. {DO_TARR_SW, SERVO_ADDR_DO_TARR_OV, 2,(unsigned char *)&servo_x.do_function_set, SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  12. {CTRL_MODE_SET, SERVO_ADDR_MODE_SET, 1,(unsigned char *)&servo_x.ctrl_mode_set, SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  13. {MOTOR_DIR_SET, SERVO_ADDR_MOTOR_DIR, 1,(unsigned char *)&servo_x.motor_dir_set, SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  14. {TARR_SOURCE_SEL,SERVO_ADDR_TARR_SEL, 14,(unsigned char *)&servo_x.tarr_source_sel,SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  15. {TARR_LIMIT12_SET, SERVO_ADDR_LIMIT_SET, 9,(unsigned char *)&servo_x.tarr_Plimit1_sel,SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  16. {SPEED_LIMIT_SET,SERVO_ADDR_SPEED_SET, 3,(unsigned char *)&servo_x.speed_plimit_set,SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  17. {TARR_BASE_SET, SERVO_ADDR_TARBASE_SET, 3,(unsigned char *)&servo_x.tarr_basis, SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  18. {ALARM_REST, SERVO_ADDR_ALARM_REST, 1,(unsigned char *)&servo_x.alarmrest, SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  19. {SERVO_RESTART, SERVO_ADDR_SERVO_RESTART,1,(unsigned char *)&servo_x.restart, SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  20. {ALARM_CODE, SERVO_ADDR_ALARM_CODE, 1,(unsigned char *)&servo_x.alarmcode, SERVO_X_COM_ID,0x03,1,50,servo_com_respone},
  21. {ENCODE_FB, SERVO_ADDR_ENCODE_FB, 2,(unsigned char *)&servo_x.encode, SERVO_X_COM_ID,0x03,1,50,servo_com_respone},
  22. {VDI_VALUE_SET, SERVO_ADDR_VDI_VALUE, 1,(unsigned char *)&servo_x.vdi_value, SERVO_X_COM_ID,0x06,0,50,servo_com_respone},
  23. {VDI_CONFIG_SET, SERVO_ADDR_VDI_USE, 12,(unsigned char *)&servo_x.vdi_use, SERVO_X_COM_ID,0x06,0,50,servo_com_respone}
  24. },
  25. {
  26. {HANDSHAKE, SERVO_ADDR_HAND, 1,(unsigned char *)&servo_y.handshake, SERVO_Y_COM_ID,0x03,0,50,servo_com_respone},
  27. {DI_MODE_SW, SERVO_ADDR_DI_MODE_SW, 2,(unsigned char *)&servo_y.di_function_set, SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  28. {DO_TARR_SW, SERVO_ADDR_DO_TARR_OV, 2,(unsigned char *)&servo_y.do_function_set, SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  29. {CTRL_MODE_SET, SERVO_ADDR_MODE_SET, 1,(unsigned char *)&servo_y.ctrl_mode_set, SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  30. {MOTOR_DIR_SET, SERVO_ADDR_MOTOR_DIR, 1,(unsigned char *)&servo_y.motor_dir_set, SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  31. {TARR_SOURCE_SEL,SERVO_ADDR_TARR_SEL, 14,(unsigned char *)&servo_y.tarr_source_sel,SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  32. {TARR_LIMIT12_SET, SERVO_ADDR_LIMIT_SET, 9,(unsigned char *)&servo_y.tarr_Plimit1_sel,SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  33. {SPEED_LIMIT_SET,SERVO_ADDR_SPEED_SET, 3,(unsigned char *)&servo_y.speed_plimit_set,SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  34. {TARR_BASE_SET, SERVO_ADDR_TARBASE_SET, 3,(unsigned char *)&servo_y.tarr_basis, SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  35. {ALARM_REST, SERVO_ADDR_ALARM_REST, 1,(unsigned char *)&servo_y.alarmrest, SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  36. {SERVO_RESTART, SERVO_ADDR_SERVO_RESTART,1,(unsigned char *)&servo_y.restart, SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  37. {ALARM_CODE, SERVO_ADDR_ALARM_CODE, 1,(unsigned char *)&servo_y.alarmcode, SERVO_Y_COM_ID,0x03,1,50,servo_com_respone},
  38. {ENCODE_FB, SERVO_ADDR_ENCODE_FB, 2,(unsigned char *)&servo_y.encode, SERVO_Y_COM_ID,0x03,1,50,servo_com_respone},
  39. {VDI_VALUE_SET, SERVO_ADDR_VDI_VALUE, 1,(unsigned char *)&servo_y.vdi_value, SERVO_Y_COM_ID,0x06,0,50,servo_com_respone},
  40. {VDI_CONFIG_SET, SERVO_ADDR_VDI_USE, 12,(unsigned char *)&servo_y.vdi_use, SERVO_Y_COM_ID,0x06,0,50,servo_com_respone}
  41. },
  42. {
  43. {HANDSHAKE, SERVO_ADDR_HAND, 1,(unsigned char *)&servo_z.handshake, SERVO_Z_COM_ID,0x03,0,50,servo_com_respone},
  44. {DI_MODE_SW, SERVO_ADDR_DI_MODE_SW, 2,(unsigned char *)&servo_z.di_function_set, SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  45. {DO_TARR_SW, SERVO_ADDR_DO_TARR_OV, 2,(unsigned char *)&servo_z.do_function_set, SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  46. {CTRL_MODE_SET, SERVO_ADDR_MODE_SET, 1,(unsigned char *)&servo_z.ctrl_mode_set, SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  47. {MOTOR_DIR_SET, SERVO_ADDR_MOTOR_DIR, 1,(unsigned char *)&servo_z.motor_dir_set, SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  48. {TARR_SOURCE_SEL,SERVO_ADDR_TARR_SEL, 14,(unsigned char *)&servo_z.tarr_source_sel,SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  49. {TARR_LIMIT12_SET, SERVO_ADDR_LIMIT_SET, 9,(unsigned char *)&servo_z.tarr_Plimit1_sel,SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  50. {SPEED_LIMIT_SET,SERVO_ADDR_SPEED_SET, 3,(unsigned char *)&servo_z.speed_plimit_set,SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  51. {TARR_BASE_SET, SERVO_ADDR_TARBASE_SET, 3,(unsigned char *)&servo_z.tarr_basis, SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  52. {ALARM_REST, SERVO_ADDR_ALARM_REST, 1,(unsigned char *)&servo_z.alarmrest, SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  53. {SERVO_RESTART, SERVO_ADDR_SERVO_RESTART,1,(unsigned char *)&servo_z.restart, SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  54. {ALARM_CODE, SERVO_ADDR_ALARM_CODE, 1,(unsigned char *)&servo_z.alarmcode, SERVO_Z_COM_ID,0x03,1,50,servo_com_respone},
  55. {ENCODE_FB, SERVO_ADDR_ENCODE_FB, 2,(unsigned char *)&servo_z.encode, SERVO_Z_COM_ID,0x03,1,50,servo_com_respone},
  56. {VDI_VALUE_SET, SERVO_ADDR_VDI_VALUE, 1,(unsigned char *)&servo_z.vdi_value, SERVO_Z_COM_ID,0x06,0,50,servo_com_respone},
  57. {VDI_CONFIG_SET, SERVO_ADDR_VDI_USE, 12,(unsigned char *)&servo_z.vdi_use, SERVO_Z_COM_ID,0x06,0,50,servo_com_respone}
  58. }
  59. };
  60. /*
  61. * 伺服通讯命令函数
  62. *
  63. * @author laz
  64. *
  65. * @param cmd
  66. */
  67. static void servo_com_cmd(unsigned short axis,unsigned char cmd)
  68. {
  69. int i,cyckletime;
  70. switch(axis)
  71. {
  72. case X_AXIS:
  73. for(i=0;i<SERVO_COM_CMD_NUMBER;i++)
  74. {
  75. if(cmd==servo_com_cmd_table[0][i].cmdcode){
  76. break;
  77. }
  78. }
  79. if(servo_com_cmd_table[0][i].dir==0) {//单次通讯
  80. servo_x.cmd_fb[servo_com_cmd_table[0][i].cmdcode]=0;
  81. //添加单次采集命令
  82. modbus_master_add_once_cmd(
  83. &master1,
  84. servo_com_cmd_table[0][i].cmdcode, //命令编码
  85. servo_com_cmd_table[0][i].id, //设备ID
  86. servo_com_cmd_table[0][i].cmd, //命令码
  87. servo_com_cmd_table[0][i].address, //10
  88. servo_com_cmd_table[0][i].number, //10
  89. servo_com_cmd_table[0][i].value, //数据存放位置
  90. servo_com_cmd_table[0][i].respone_timeout, //50ms超时时间(不包括通讯必须时间)
  91. servo_com_cmd_table[0][i].servo_on_respone //不需要回调函数
  92. );
  93. }
  94. else
  95. {
  96. if(cmd==ENCODE_FB)cyckletime=30;
  97. else cyckletime=1000;
  98. //添加循环采集命令
  99. modbus_master_add_period_cmd(
  100. &master1,
  101. servo_com_cmd_table[0][i].cmdcode, //命令编码
  102. servo_com_cmd_table[0][i].id, //设备ID
  103. servo_com_cmd_table[0][i].cmd, //命令码
  104. servo_com_cmd_table[0][i].address, //10
  105. servo_com_cmd_table[0][i].number, //10
  106. servo_com_cmd_table[0][i].value, //数据存放位置
  107. servo_com_cmd_table[0][i].respone_timeout, //50ms超时时间(不包括通讯必须时间)
  108. servo_com_cmd_table[0][i].servo_on_respone,//不需要回调函数
  109. cyckletime //1000ms采集一次
  110. );
  111. }
  112. break;
  113. case Y_AXIS:
  114. for(i=0;i<SERVO_COM_CMD_NUMBER;i++)
  115. {
  116. if(cmd==servo_com_cmd_table[1][i].cmdcode){
  117. break;
  118. }
  119. }
  120. if(servo_com_cmd_table[1][i].dir==0) {//单次通讯
  121. servo_y.cmd_fb[servo_com_cmd_table[1][i].cmdcode]=0;
  122. //添加单次采集命令
  123. modbus_master_add_once_cmd(
  124. &master1,
  125. servo_com_cmd_table[1][i].cmdcode, //命令编码
  126. servo_com_cmd_table[1][i].id, //设备ID
  127. servo_com_cmd_table[1][i].cmd, //命令码
  128. servo_com_cmd_table[1][i].address, //10
  129. servo_com_cmd_table[1][i].number, //10
  130. servo_com_cmd_table[1][i].value, //数据存放位置
  131. servo_com_cmd_table[1][i].respone_timeout, //50ms超时时间(不包括通讯必须时间)
  132. servo_com_cmd_table[1][i].servo_on_respone //不需要回调函数
  133. );
  134. }
  135. else
  136. {
  137. if(cmd==ENCODE_FB)cyckletime=100;
  138. else cyckletime=1000;
  139. //添加循环采集命令
  140. modbus_master_add_period_cmd(
  141. &master1,
  142. servo_com_cmd_table[1][i].cmdcode, //命令编码
  143. servo_com_cmd_table[1][i].id, //设备ID
  144. servo_com_cmd_table[1][i].cmd, //命令码
  145. servo_com_cmd_table[1][i].address, //10
  146. servo_com_cmd_table[1][i].number, //10
  147. servo_com_cmd_table[1][i].value, //数据存放位置
  148. servo_com_cmd_table[1][i].respone_timeout, //50ms超时时间(不包括通讯必须时间)
  149. servo_com_cmd_table[1][i].servo_on_respone,//不需要回调函数
  150. cyckletime //1000ms采集一次
  151. );
  152. }
  153. break;
  154. case Z_AXIS:
  155. for(i=0;i<SERVO_COM_CMD_NUMBER;i++)
  156. {
  157. if(cmd==servo_com_cmd_table[2][i].cmdcode){
  158. break;
  159. }
  160. }
  161. if(servo_com_cmd_table[2][i].dir==0) {//单次通讯
  162. servo_y.cmd_fb[servo_com_cmd_table[2][i].cmdcode]=0;
  163. //添加单次采集命令
  164. modbus_master_add_once_cmd(
  165. &master1,
  166. servo_com_cmd_table[2][i].cmdcode, //命令编码
  167. servo_com_cmd_table[2][i].id, //设备ID
  168. servo_com_cmd_table[2][i].cmd, //命令码
  169. servo_com_cmd_table[2][i].address, //10
  170. servo_com_cmd_table[2][i].number, //10
  171. servo_com_cmd_table[2][i].value, //数据存放位置
  172. servo_com_cmd_table[2][i].respone_timeout, //50ms超时时间(不包括通讯必须时间)
  173. servo_com_cmd_table[2][i].servo_on_respone //不需要回调函数
  174. );
  175. }
  176. else
  177. {
  178. if(cmd==ENCODE_FB)cyckletime=100;
  179. else cyckletime=1000;
  180. //添加循环采集命令
  181. modbus_master_add_period_cmd(
  182. &master1,
  183. servo_com_cmd_table[2][i].cmdcode, //命令编码
  184. servo_com_cmd_table[2][i].id, //设备ID
  185. servo_com_cmd_table[2][i].cmd, //命令码
  186. servo_com_cmd_table[2][i].address, //10
  187. servo_com_cmd_table[2][i].number, //10
  188. servo_com_cmd_table[2][i].value, //数据存放位置
  189. servo_com_cmd_table[2][i].respone_timeout, //50ms超时时间(不包括通讯必须时间)
  190. servo_com_cmd_table[2][i].servo_on_respone,//不需要回调函数
  191. cyckletime //1000ms采集一次
  192. );
  193. }
  194. break;
  195. }
  196. }
  197. /*
  198. * 伺服通讯,故障清除函数
  199. *
  200. * @author laz
  201. *
  202. * @param aixs
  203. */
  204. int clr_com_servo_alarm(unsigned short axis)
  205. {
  206. static unsigned long clr_com_servo_x_delay=0;
  207. static unsigned long clr_com_servo_y_delay=0;
  208. static unsigned long clr_com_servo_z_delay=0;
  209. switch(axis)
  210. {
  211. case X_AXIS:
  212. if(dwTickCount>=clr_com_servo_x_delay)
  213. {
  214. servo_x.alarmrest=1;
  215. clr_com_servo_x_delay=dwTickCount+1000;
  216. servo_com_cmd(axis,ALARM_REST);
  217. }
  218. break;
  219. case Y_AXIS:
  220. if(dwTickCount>=clr_com_servo_y_delay)
  221. {
  222. servo_y.alarmrest=1;
  223. servo_com_cmd(axis,ALARM_REST);
  224. clr_com_servo_y_delay=dwTickCount+1000;
  225. }
  226. break;
  227. case Z_AXIS:
  228. if(dwTickCount>=clr_com_servo_z_delay)
  229. {
  230. servo_z.alarmrest=1;
  231. servo_com_cmd(axis,ALARM_REST);
  232. clr_com_servo_z_delay=dwTickCount+1000;
  233. }
  234. break;
  235. }
  236. return 1;
  237. }
  238. /*
  239. * 伺服通讯,设置细分,设置周长
  240. *
  241. * @author laz
  242. *
  243. * @param aixs
  244. */
  245. void set_com_servo_param(unsigned short axis,unsigned short cycle_pluse,unsigned short cycle_length)
  246. {
  247. switch(axis)
  248. {
  249. case X_AXIS:
  250. servo_x.cycle_pulse=cycle_pluse;
  251. servo_x.cycle_length=cycle_length;
  252. break;
  253. case Y_AXIS:
  254. servo_y.cycle_pulse=cycle_pluse;
  255. servo_y.cycle_length=cycle_length;
  256. break;
  257. case Z_AXIS:
  258. servo_z.cycle_pulse=cycle_pluse;
  259. servo_z.cycle_length=cycle_length;
  260. break;
  261. }
  262. }
  263. /*
  264. * 伺服通讯,设置伺服告警极性
  265. *
  266. * @author laz
  267. *
  268. * @param aixs
  269. */
  270. void set_com_alarm_reverse(unsigned short axis,int value)
  271. {
  272. switch(axis)
  273. {
  274. case X_AXIS:
  275. servo_x.alarm_Reverse=value;
  276. break;
  277. case Y_AXIS:
  278. servo_y.alarm_Reverse=value;
  279. break;
  280. case Z_AXIS:
  281. servo_z.alarm_Reverse=value;
  282. break;
  283. }
  284. }
  285. /*
  286. * 伺服通讯,获取伺服故障代码
  287. *
  288. * @author laz
  289. *
  290. * @param aixs
  291. */
  292. int get_alarm_code(unsigned short axis)
  293. {
  294. switch(axis)
  295. {
  296. case X_AXIS:
  297. return (servo_x.alarmcode);
  298. break;
  299. case Y_AXIS:
  300. return (servo_y.alarmcode);
  301. break;
  302. case Z_AXIS:
  303. return (servo_z.alarmcode);
  304. break;
  305. }
  306. return 0;
  307. }
  308. /*
  309. * 伺服通讯,编码转换长度计算
  310. *
  311. * @author laz
  312. *
  313. * @param encode 伺服编码
  314. * gearratio,齿轮比
  315. */
  316. static long encode_to_length(long encode,unsigned short cycle_length)
  317. {
  318. float pos;
  319. pos=encode;
  320. pos=(float)(pos/ENCODE_17BIT);
  321. pos*= cycle_length;
  322. return (long)(pos);
  323. }
  324. /*
  325. * 伺服通讯,位置设置值
  326. *
  327. * @author laz
  328. *
  329. * @param axis
  330. * pos,
  331. */
  332. int axis_set_com_pos(unsigned short axis,long pos )
  333. {
  334. switch(axis)
  335. {
  336. case X_AXIS:
  337. servo_x.precode=servo_x.encode-pos;
  338. break;
  339. case Y_AXIS:
  340. servo_y.precode=servo_y.encode-pos;
  341. break;
  342. case Z_AXIS:
  343. servo_z.precode=servo_z.encode-pos;
  344. break;
  345. }
  346. return 1;
  347. }
  348. /*
  349. * 伺服通讯,长度值
  350. *
  351. * @author laz
  352. *
  353. * @param axis
  354. */
  355. static long axis_get_com_pos(unsigned short axis)
  356. {
  357. switch(axis)
  358. {
  359. case X_AXIS:
  360. if(servo_x.encode>=servo_x.precode)
  361. return(encode_to_length(servo_x.encode-servo_x.precode,servo_x.cycle_length));
  362. else
  363. return(encode_to_length(servo_x.precode-servo_x.encode,servo_x.cycle_length));
  364. break;
  365. case Y_AXIS:
  366. if(servo_y.encode>=servo_y.precode)
  367. return(encode_to_length(servo_y.encode-servo_y.precode,servo_y.cycle_length));
  368. else
  369. return(encode_to_length(servo_y.precode-servo_y.encode,servo_y.cycle_length));
  370. break;
  371. case Z_AXIS:
  372. if(servo_z.encode>=servo_z.precode)
  373. return(encode_to_length(servo_z.encode-servo_z.precode,servo_z.cycle_length));
  374. else
  375. return(encode_to_length(servo_z.precode-servo_z.encode,servo_z.cycle_length));
  376. break;
  377. }
  378. return 0;
  379. }
  380. /*
  381. * 伺服通讯,//频率转换为转速
  382. *
  383. * @author laz
  384. *
  385. * @param freq 伺服的频率
  386. * rerurn 伺服的转速
  387. */
  388. static unsigned short freq_to_speed(unsigned short freq,unsigned short cycle_pulse)
  389. {
  390. if(freq==0)return 1;
  391. if(cycle_pulse==0)return 1;
  392. return(freq*60/cycle_pulse+1);
  393. }
  394. /*
  395. * 伺服通讯,读取编码次数
  396. *
  397. * @author laz
  398. *
  399. * @param axis
  400. * times 读取次数
  401. */
  402. void read_encode_value_cmd(unsigned short axis,unsigned short times)
  403. {
  404. int i=0;
  405. switch(axis)
  406. {
  407. case X_AXIS:
  408. //servo_x.cmd_fb[ENCODE_FB]=0;
  409. for(i=0;i<times;i++)servo_com_cmd(axis,ENCODE_FB);
  410. break;
  411. case Y_AXIS:
  412. //servo_y.cmd_fb[ENCODE_FB]=0;
  413. for(i=0;i<times;i++)servo_com_cmd(axis,ENCODE_FB);
  414. break;
  415. case Z_AXIS:
  416. //servo_y.cmd_fb[ENCODE_FB]=0;
  417. for(i=0;i<times;i++)servo_com_cmd(axis,ENCODE_FB);
  418. break;
  419. }
  420. }
  421. /*
  422. * 伺服通讯,读取长度值
  423. *
  424. * @author laz
  425. *
  426. * @param axis
  427. */
  428. long get_encode_value(unsigned short axis)
  429. {
  430. switch(axis)
  431. {
  432. case X_AXIS:
  433. return axis_get_com_pos(axis);
  434. break;
  435. case Y_AXIS:
  436. return axis_get_com_pos(axis);
  437. break;
  438. case Z_AXIS:
  439. return axis_get_com_pos(axis);
  440. break;
  441. }
  442. return 0;
  443. }
  444. /*
  445. * 伺服通讯,伺服轴通讯回调函数
  446. *
  447. * @author laz
  448. *
  449. *
  450. */
  451. static void servo_axis_com_respone(unsigned short axis,servo_param_t *servo_axis,modbus_master_cmd_t *cmd, unsigned char *respone, int length)
  452. {
  453. switch(cmd->cmdcode)
  454. {
  455. case HANDSHAKE:
  456. servo_axis->cmd_fb[HANDSHAKE]=cmd->cmdcode;
  457. break;
  458. case DI_MODE_SW:
  459. servo_axis->cmd_fb[DI_MODE_SW]=cmd->cmdcode;
  460. break;
  461. case DO_TARR_SW:
  462. servo_axis->cmd_fb[DO_TARR_SW]=cmd->cmdcode;
  463. break;
  464. case CTRL_MODE_SET:
  465. servo_axis->cmd_fb[CTRL_MODE_SET]=cmd->cmdcode;
  466. break;
  467. case MOTOR_DIR_SET:
  468. servo_axis->cmd_fb[MOTOR_DIR_SET]=cmd->cmdcode;
  469. break;
  470. case TARR_SOURCE_SEL:
  471. servo_axis->cmd_fb[TARR_SOURCE_SEL]=cmd->cmdcode;
  472. break;
  473. case TARR_LIMIT12_SET:
  474. servo_axis->set_tarr_OKflag++;
  475. servo_axis->cmd_fb[TARR_LIMIT12_SET]=cmd->cmdcode;
  476. break;
  477. case SPEED_LIMIT_SET:
  478. servo_axis->set_tarr_OKflag++;
  479. servo_axis->cmd_fb[SPEED_LIMIT_SET]=cmd->cmdcode;
  480. break;
  481. case TARR_BASE_SET:
  482. servo_axis->set_tarr_OKflag++;
  483. servo_axis->cmd_fb[TARR_BASE_SET]=cmd->cmdcode;
  484. break;
  485. case ALARM_REST:
  486. servo_axis->cmd_fb[ALARM_REST]=cmd->cmdcode;
  487. break;
  488. case SERVO_RESTART:
  489. servo_axis->cmd_fb[SERVO_RESTART]=cmd->cmdcode;
  490. break;
  491. case ALARM_CODE:
  492. servo_axis->cmd_fb[ALARM_CODE]=cmd->cmdcode;
  493. if(servo_axis->alarmcode>0){
  494. if(servo_axis->alarm_Reverse)servo_axis->alarm_flag=0;else servo_axis->alarm_flag=1;
  495. }
  496. else {
  497. if(servo_axis->alarm_Reverse)servo_axis->alarm_flag=1;else servo_axis->alarm_flag=0;
  498. }
  499. break;
  500. case ENCODE_FB:
  501. servo_axis->cmd_fb[ENCODE_FB]=cmd->cmdcode;
  502. get_encode_value(axis);
  503. break;
  504. default:
  505. break;
  506. }
  507. }
  508. static int servo_com_get_alarm(unsigned short axis)
  509. {
  510. switch(axis)
  511. {
  512. case X_AXIS:
  513. axis_x_com_alarm=servo_x.alarm_flag;
  514. break;
  515. case Y_AXIS:
  516. axis_y_com_alarm=servo_y.alarm_flag;
  517. break;
  518. case Z_AXIS:
  519. axis_z_com_alarm=servo_z.alarm_flag;
  520. break;
  521. }
  522. return 1;
  523. }
  524. /*
  525. * 伺服通讯,通讯回调函数
  526. *
  527. * @author laz
  528. *
  529. *
  530. */
  531. void servo_com_respone(modbus_master_cmd_t *cmd, unsigned char *respone, int length)
  532. {
  533. #if X_USERING_TARR == 1
  534. if(cmd->id==SERVO_X_COM_ID)servo_axis_com_respone(X_AXIS,&servo_x,cmd,respone,length);
  535. #endif
  536. #if Y_USERING_TARR == 1
  537. if(cmd->id==SERVO_Y_COM_ID)servo_axis_com_respone(Y_AXIS,&servo_y,cmd,respone,length);
  538. #endif
  539. #if Z_USERING_TARR == 1
  540. if(cmd->id==SERVO_Z_COM_ID)servo_axis_com_respone(Z_AXIS,&servo_z,cmd,respone,length);
  541. #endif
  542. }
  543. //设置位置+转矩模式下的转矩配置
  544. int set_servo_postotarr_limit(unsigned short axis,int tarr,
  545. unsigned short P_freq,unsigned short N_freq,
  546. unsigned short dir)
  547. {
  548. switch(axis)
  549. {
  550. case X_AXIS:
  551. //配置扭矩模式下的速度限制、扭矩限制、扭矩基准值
  552. servo_x.speed_plimit_set=freq_to_speed(P_freq,servo_x.cycle_pulse);
  553. servo_x.speed_nlimit_set=freq_to_speed(N_freq,servo_x.cycle_pulse);
  554. if(tarr<100)tarr=100;
  555. if(dir)servo_x.tarr_limit_set=-tarr;
  556. else servo_x.tarr_limit_set=tarr;
  557. servo_com_cmd(axis,SPEED_LIMIT_SET);
  558. //servo_x.cmd_fb[SPEED_LIMIT_SET]=0;//清除反馈命令
  559. //转矩的基准值设置
  560. servo_x.tarr_basis=tarr-50;
  561. servo_x.tarr_basis_up=50;
  562. servo_x.tarr_basis_dw=30;
  563. servo_com_cmd(axis,TARR_BASE_SET);
  564. //servo_x.cmd_fb[TARR_BASE_SET]=0;//清除反馈命令
  565. servo_x.set_tarr_OKflag=0;
  566. sw_timer_start(&servo_x.timer, 1, 0);//1S超时
  567. break;
  568. case Y_AXIS:
  569. //配置扭矩模式下的速度限制、扭矩限制、扭矩基准值
  570. servo_y.speed_plimit_set=freq_to_speed(P_freq,servo_y.cycle_pulse);
  571. servo_y.speed_nlimit_set=freq_to_speed(N_freq,servo_y.cycle_pulse);
  572. if(tarr<100)tarr=100;
  573. if(dir)servo_y.tarr_limit_set=-tarr;
  574. else servo_y.tarr_limit_set=tarr;
  575. servo_com_cmd(axis,SPEED_LIMIT_SET);
  576. //servo_y.cmd_fb[SPEED_LIMIT_SET]=0;//清除反馈命令
  577. //转矩的基准值设置
  578. servo_y.tarr_basis=tarr-50;
  579. servo_y.tarr_basis_up=50;
  580. servo_y.tarr_basis_dw=30;
  581. servo_com_cmd(axis,TARR_BASE_SET);
  582. //servo_y.cmd_fb[TARR_BASE_SET]=0;//清除反馈命令
  583. servo_y.set_tarr_OKflag=0;
  584. sw_timer_start(&servo_y.timer, 1, 0);//1S超时
  585. break;
  586. case Z_AXIS:
  587. //配置扭矩模式下的速度限制、扭矩限制、扭矩基准值
  588. servo_z.speed_plimit_set=freq_to_speed(P_freq,servo_z.cycle_pulse);
  589. servo_z.speed_nlimit_set=freq_to_speed(N_freq,servo_z.cycle_pulse);
  590. if(tarr<100)tarr=100;
  591. if(dir)servo_z.tarr_limit_set=-tarr;
  592. else servo_z.tarr_limit_set=tarr;
  593. servo_com_cmd(axis,SPEED_LIMIT_SET);
  594. //servo_z.cmd_fb[SPEED_LIMIT_SET]=0;//清除反馈命令
  595. //转矩的基准值设置
  596. servo_z.tarr_basis=tarr-50;
  597. servo_z.tarr_basis_up=50;
  598. servo_z.tarr_basis_dw=30;
  599. servo_com_cmd(axis,TARR_BASE_SET);
  600. //servo_z.cmd_fb[TARR_BASE_SET]=0;//清除反馈命令
  601. servo_z.set_tarr_OKflag=0;
  602. sw_timer_start(&servo_z.timer, 1, 0);//1S超时
  603. break;
  604. }
  605. return 1;
  606. }
  607. //转矩模式下的扭矩配置
  608. //设置位置+转矩模式下的转矩配置
  609. int set_servo_tarr_limit(unsigned short axis,int maxtarr,int mintarr,
  610. unsigned short P_freq,unsigned short N_freq,
  611. unsigned short dir)
  612. {
  613. switch(axis)
  614. {
  615. case X_AXIS:
  616. if(maxtarr<400)maxtarr=400;
  617. if(maxtarr>=2000)maxtarr=2000;
  618. //扭矩1,扭矩2的配置
  619. servo_x.tarr_Plimit1_sel=mintarr;
  620. servo_x.tarr_Nlimit1_sel=mintarr;
  621. servo_x.tarr_Plimit2_sel=maxtarr;
  622. servo_x.tarr_Nlimit2_sel=maxtarr;
  623. servo_x.speed_limit_sel=0;
  624. servo_x.speed_ai_sel=1;
  625. //配置扭矩模式下的速度限制、扭矩基准值
  626. servo_x.speed_plimit_set=freq_to_speed(P_freq,servo_x.cycle_pulse);
  627. servo_x.speed_nlimit_set=freq_to_speed(N_freq,servo_x.cycle_pulse);
  628. if(dir)servo_x.tarr_limit_set=-(maxtarr+100);
  629. else servo_x.tarr_limit_set=(maxtarr+100);
  630. servo_com_cmd(axis,TARR_LIMIT12_SET);
  631. //servo_x.cmd_fb[TARR_LIMIT12_SET]=0;//清除反馈命令
  632. //转矩的基准值设置
  633. servo_x.tarr_basis=mintarr-50;
  634. servo_x.tarr_basis_up=50;
  635. servo_x.tarr_basis_dw=30;
  636. servo_com_cmd(axis,TARR_BASE_SET);
  637. //servo_x.cmd_fb[TARR_BASE_SET]=0;//清除反馈命令
  638. servo_x.set_tarr_OKflag=0;
  639. sw_timer_start(&servo_x.timer, 1, 0);//1S超时
  640. break;
  641. case Y_AXIS:
  642. if(maxtarr<400)maxtarr=400;
  643. if(maxtarr>=2000)maxtarr=2000;
  644. //扭矩1,扭矩2的配置
  645. servo_y.tarr_Plimit1_sel=mintarr;
  646. servo_y.tarr_Nlimit1_sel=mintarr;
  647. servo_y.tarr_Plimit2_sel=maxtarr;
  648. servo_y.tarr_Nlimit2_sel=maxtarr;
  649. servo_y.speed_limit_sel=0;
  650. servo_y.speed_ai_sel=1;
  651. //配置扭矩模式下的速度限制、扭矩基准值
  652. servo_y.speed_plimit_set=freq_to_speed(P_freq,servo_y.cycle_pulse);
  653. servo_y.speed_nlimit_set=freq_to_speed(N_freq,servo_y.cycle_pulse);
  654. if(dir)servo_y.tarr_limit_set=-(maxtarr+100);
  655. else servo_y.tarr_limit_set=(maxtarr+100);
  656. servo_com_cmd(axis,TARR_LIMIT12_SET);
  657. //servo_y.cmd_fb[TARR_LIMIT12_SET]=0;//清除反馈命令
  658. //转矩的基准值设置
  659. servo_y.tarr_basis=mintarr-50;
  660. servo_y.tarr_basis_up=50;
  661. servo_y.tarr_basis_dw=30;
  662. servo_com_cmd(axis,TARR_BASE_SET);
  663. //servo_y.cmd_fb[TARR_BASE_SET]=0;//清除反馈命令
  664. servo_y.set_tarr_OKflag=0;
  665. sw_timer_start(&servo_y.timer, 1, 0);//1S超时
  666. break;
  667. case Z_AXIS:
  668. if(maxtarr<400)maxtarr=400;
  669. if(maxtarr>=2000)maxtarr=2000;
  670. //扭矩1,扭矩2的配置
  671. servo_z.tarr_Plimit1_sel=mintarr;
  672. servo_z.tarr_Nlimit1_sel=mintarr;
  673. servo_z.tarr_Plimit2_sel=maxtarr;
  674. servo_z.tarr_Nlimit2_sel=maxtarr;
  675. servo_z.speed_limit_sel=0;
  676. servo_z.speed_ai_sel=1;
  677. //配置扭矩模式下的速度限制、扭矩基准值
  678. servo_z.speed_plimit_set=freq_to_speed(P_freq,servo_z.cycle_pulse);
  679. servo_z.speed_nlimit_set=freq_to_speed(N_freq,servo_z.cycle_pulse);
  680. if(dir)servo_z.tarr_limit_set=-(maxtarr+100);
  681. else servo_z.tarr_limit_set=(maxtarr+100);
  682. servo_com_cmd(axis,TARR_LIMIT12_SET);
  683. //servo_z.cmd_fb[TARR_LIMIT12_SET]=0;//清除反馈命令
  684. //转矩的基准值设置
  685. servo_z.tarr_basis=mintarr-50;
  686. servo_z.tarr_basis_up=50;
  687. servo_z.tarr_basis_dw=30;
  688. servo_com_cmd(axis,TARR_BASE_SET);
  689. //servo_z.cmd_fb[TARR_BASE_SET]=0;//清除反馈命令
  690. servo_z.set_tarr_OKflag=0;
  691. sw_timer_start(&servo_z.timer, 1, 0);//1S超时
  692. break;
  693. }
  694. return 1;
  695. }
  696. /**
  697. * 获取转矩配置是否成功
  698. *
  699. * @author laz
  700. *
  701. * @param axis
  702. */
  703. int get_tarr_set(unsigned short axis)
  704. {
  705. switch(axis)
  706. {
  707. case X_AXIS:
  708. if(servo_x.set_tarr_OKflag==2)return 2;//配置成功
  709. if(sw_timer_expire(&servo_x.timer))return 0;//配置超时
  710. else return 1;//配置中
  711. break;
  712. case Y_AXIS:
  713. if(servo_y.set_tarr_OKflag==2)return 2;//配置成功
  714. if(sw_timer_expire(&servo_y.timer))return 0;//配置超时
  715. else return 1;//配置中
  716. break;
  717. case Z_AXIS:
  718. if(servo_z.set_tarr_OKflag==2)return 2;//配置成功
  719. if(sw_timer_expire(&servo_z.timer))return 0;//配置超时
  720. else return 1;//配置中
  721. break;
  722. }
  723. return 0;
  724. }
  725. //设置转矩模式下的X配置初始化
  726. static int set_ctrl_tarr_axis_init(unsigned short axis,servo_param_t *servo_axis,unsigned short mode)
  727. {
  728. switch(mode)
  729. {
  730. case 0:
  731. servo_axis->tarr_source_sel=0;//来源A
  732. servo_axis->tarr_modea_sel=0;
  733. servo_axis->tarr_modeb_sel=0;
  734. servo_axis->tarr_limit_sel=0;
  735. servo_axis->tarr_ai_sel=1;
  736. servo_axis->tarr_Plimit1_sel=3000;
  737. servo_axis->tarr_Nlimit1_sel=3000;
  738. servo_axis->tarr_Plimit2_sel=3000;
  739. servo_axis->tarr_Nlimit2_sel=3000;
  740. servo_axis->speed_limit_sel=0;
  741. servo_axis->speed_ai_sel=1;
  742. servo_axis->speed_plimit_set=200;
  743. servo_axis->speed_nlimit_set=100;
  744. servo_axis->tarr_limit_set=50;
  745. servo_com_cmd(axis,TARR_SOURCE_SEL);
  746. //转矩的基准值设置
  747. servo_axis->tarr_basis=0;
  748. servo_axis->tarr_basis_up=200;
  749. servo_axis->tarr_basis_dw=0;
  750. servo_com_cmd(axis,TARR_BASE_SET);
  751. //servo_axis->cmd_fb[TARR_SOURCE_SEL]=0;//清除反馈命令
  752. break;
  753. case 1:
  754. servo_axis->tarr_source_sel=3;//来源A
  755. servo_axis->tarr_modea_sel=0;
  756. servo_axis->tarr_modeb_sel=0;
  757. servo_axis->tarr_limit_sel=2;
  758. servo_axis->tarr_ai_sel=1;
  759. servo_axis->tarr_Plimit1_sel=200;
  760. servo_axis->tarr_Nlimit1_sel=200;
  761. servo_axis->tarr_Plimit2_sel=1000;
  762. servo_axis->tarr_Nlimit2_sel=1000;
  763. servo_axis->speed_limit_sel=0;
  764. servo_axis->speed_ai_sel=1;
  765. servo_axis->speed_plimit_set=200;
  766. servo_axis->speed_nlimit_set=100;
  767. servo_axis->tarr_limit_set=1000;
  768. servo_com_cmd(axis,TARR_SOURCE_SEL);
  769. //转矩的基准值设置
  770. servo_axis->tarr_basis=0;
  771. servo_axis->tarr_basis_up=200;
  772. servo_axis->tarr_basis_dw=0;
  773. servo_com_cmd(axis,TARR_BASE_SET);
  774. //servo_axis->cmd_fb[TARR_SOURCE_SEL]=0;//清除反馈命令
  775. break;
  776. }
  777. return 1;
  778. }
  779. //设置转矩模式下的配置
  780. static int set_ctrl_tarr_config(unsigned short axis,unsigned short mode)
  781. {
  782. switch(axis)
  783. {
  784. case X_AXIS:
  785. set_ctrl_tarr_axis_init(axis,&servo_x,mode);
  786. break;
  787. case Y_AXIS:
  788. set_ctrl_tarr_axis_init(axis,&servo_y,mode);
  789. break;
  790. case Z_AXIS:
  791. set_ctrl_tarr_axis_init(axis,&servo_z,mode);
  792. break;
  793. }
  794. return 1;
  795. }
  796. //设置伺服运行模式
  797. //0位置+转矩模式,转矩改变由通讯方式
  798. //1转矩模式,采用转矩切换方式
  799. static void set_servo_axis_mode(unsigned short axis,servo_param_t *servo_axis,unsigned short mode)
  800. {
  801. //SetEn(axis, MOTOR_EN);//停机关使能
  802. switch(mode)
  803. {
  804. case 0:
  805. //配置DI5功能为模式1切换和极性
  806. servo_axis->di_function_set=DI_FUNCTION_POSTARRSEL;
  807. servo_axis->di_polarity= DI_NCLOSE;
  808. servo_com_cmd(axis,DI_MODE_SW);
  809. //servo_axis->cmd_fb[DI_MODE_SW]=0;//清除反馈命令
  810. //配置DO3为转矩到达输出和极性
  811. servo_axis->do_function_set=DO_FUNCTION_VALUE;
  812. servo_axis->do_polarity= DI_NOPEN;
  813. servo_com_cmd(axis,DO_TARR_SW);
  814. //servo_axis->cmd_fb[DO_TARR_SW]=0;//清除反馈命令
  815. //配置控制模式
  816. servo_axis->ctrl_mode_set=CTRL_POSTOTARR_MODE;
  817. servo_com_cmd(axis,CTRL_MODE_SET);
  818. //servo_axis->cmd_fb[CTRL_MODE_SET]=0;//清除反馈命令
  819. //配置转矩的方式:
  820. set_ctrl_tarr_config(axis,mode);
  821. //配置扭矩模式下的速度限制、扭矩限制、扭矩基准值
  822. //set_servo_postotarr_limit(axis,550,10000,10000,SERVO_TARR_CCW);
  823. //配置超时设置
  824. sw_timer_start(&servo_axis->timer, 2, 0);//2S超时
  825. break;
  826. case 1:
  827. //配置DI5功能为模式1切换和极性
  828. servo_axis->di_function_set=DI_FUNCTION_TARRSEL;
  829. servo_axis->di_polarity= DI_NOPEN;
  830. servo_com_cmd(axis,DI_MODE_SW);
  831. //servo_axis->cmd_fb[DI_MODE_SW]=0;//清除反馈命令
  832. //配置DO3为转矩到达输出和极性
  833. servo_axis->do_function_set=DO_FUNCTION_VALUE;
  834. servo_axis->do_polarity= DI_NOPEN;
  835. servo_com_cmd(axis,DO_TARR_SW);
  836. //servo_axis->cmd_fb[DO_TARR_SW]=0;//清除反馈命令
  837. //配置控制模式
  838. servo_axis->ctrl_mode_set=CTRL_TARR_MODE;
  839. servo_com_cmd(axis,CTRL_MODE_SET);
  840. //servo_axis->cmd_fb[CTRL_MODE_SET]=0;//清除反馈命令
  841. //配置转矩的方式:
  842. set_ctrl_tarr_config(axis,mode);
  843. //配置扭矩模式下的速度限制、扭矩限制、扭矩基准值
  844. //set_servo_tarr_limit(axis,1000,1000,20000,10000,SERVO_TARR_CCW);
  845. //配置超时设置
  846. sw_timer_start(&servo_axis->timer, 2, 0);//2S超时
  847. break;
  848. }
  849. }
  850. //设置伺服运行模式
  851. //0位置+转矩模式,转矩改变由通讯方式
  852. //1转矩模式,采用转矩切换方式
  853. static void set_servo_mode(unsigned short axis,unsigned short mode)
  854. {
  855. switch(axis)
  856. {
  857. case X_AXIS:
  858. set_servo_axis_mode(X_AXIS,&servo_x,mode);
  859. break;
  860. case Y_AXIS:
  861. set_servo_axis_mode(Y_AXIS,&servo_y,mode);
  862. break;
  863. case Z_AXIS:
  864. set_servo_axis_mode(Z_AXIS,&servo_z,mode);
  865. break;
  866. }
  867. }
  868. //设置伺服运行模式使能
  869. void set_servo_runmode(unsigned short axis,unsigned short mode)
  870. {
  871. switch(axis)
  872. {
  873. case X_AXIS:
  874. servo_x.run_mode=mode;
  875. break;
  876. case Y_AXIS:
  877. servo_y.run_mode=mode;
  878. break;
  879. case Z_AXIS:
  880. servo_z.run_mode=mode;
  881. break;
  882. }
  883. }
  884. //读取编码器使能,
  885. void ReadServoEncode_Enable(unsigned short axis,unsigned short en)
  886. {
  887. switch(axis)
  888. {
  889. case X_AXIS:
  890. servo_x.read_encode_en=en;
  891. break;
  892. case Y_AXIS:
  893. servo_y.read_encode_en=en;
  894. break;
  895. case Z_AXIS:
  896. servo_z.read_encode_en=en;
  897. break;
  898. }
  899. }
  900. //读取编码器使能,
  901. void Set_read_encode(unsigned short axis)
  902. {
  903. switch(axis)
  904. {
  905. case X_AXIS:
  906. if(servo_x.read_encode_en)servo_com_cmd(axis,ENCODE_FB);
  907. break;
  908. case Y_AXIS:
  909. if(servo_y.read_encode_en)servo_com_cmd(axis,ENCODE_FB);
  910. break;
  911. case Z_AXIS:
  912. if(servo_z.read_encode_en)servo_com_cmd(axis,ENCODE_FB);
  913. break;
  914. }
  915. }
  916. //读取故障使能,
  917. void ReadServoAlarm_Enable(unsigned short axis,unsigned short en)
  918. {
  919. switch(axis)
  920. {
  921. case X_AXIS:
  922. servo_x.read_alarm_en=en;
  923. break;
  924. case Y_AXIS:
  925. servo_y.read_alarm_en=en;
  926. break;
  927. case Z_AXIS:
  928. servo_z.read_alarm_en=en;
  929. break;
  930. }
  931. }
  932. //读取编码器使能,
  933. void Set_read_alarm(unsigned short axis)
  934. {
  935. switch(axis)
  936. {
  937. case X_AXIS:
  938. if(servo_x.read_alarm_en)servo_com_cmd(axis,ALARM_CODE);
  939. break;
  940. case Y_AXIS:
  941. if(servo_y.read_alarm_en)servo_com_cmd(axis,ALARM_CODE);
  942. break;
  943. case Z_AXIS:
  944. if(servo_z.read_alarm_en)servo_com_cmd(axis,ALARM_CODE);
  945. break;
  946. }
  947. }
  948. //
  949. /**
  950. * 伺服通讯总线虚拟IO配置
  951. *
  952. * @author laz
  953. *
  954. * @param cmd
  955. */
  956. static int vdi_config_init(unsigned short axis)
  957. {
  958. switch(axis)
  959. {
  960. case X_AXIS:
  961. servo_x.vdi_use=1;
  962. servo_x.vdi_init=0;
  963. servo_x.vdo_use=0;
  964. servo_x.vdo_init=0;
  965. servo_x.vdi1_function_set=VDI_SON_FUNCTION;
  966. servo_x.vdi1_polarity=DI_NOPEN;
  967. servo_x.vdi2_function_set=VDI_CLRENCODE_FUNCTION;
  968. servo_x.vdi2_polarity=DI_NOPEN;
  969. servo_com_cmd(axis,VDI_CONFIG_SET);
  970. break;
  971. case Y_AXIS:
  972. servo_y.vdi_use=1;
  973. servo_y.vdi_init=0;
  974. servo_y.vdo_use=0;
  975. servo_y.vdo_init=0;
  976. servo_y.vdi1_function_set=VDI_SON_FUNCTION;
  977. servo_y.vdi1_polarity=DI_NOPEN;
  978. servo_y.vdi2_function_set=VDI_CLRENCODE_FUNCTION;
  979. servo_y.vdi2_polarity=DI_NOPEN;
  980. servo_com_cmd(axis,VDI_CONFIG_SET);
  981. break;
  982. case Z_AXIS:
  983. servo_z.vdi_use=1;
  984. servo_z.vdi_init=0;
  985. servo_z.vdo_use=0;
  986. servo_z.vdo_init=0;
  987. servo_z.vdi1_function_set=VDI_SON_FUNCTION;
  988. servo_z.vdi1_polarity=DI_NOPEN;
  989. servo_z.vdi2_function_set=VDI_CLRENCODE_FUNCTION;
  990. servo_z.vdi2_polarity=DI_NOPEN;
  991. servo_com_cmd(axis,VDI_CONFIG_SET);
  992. break;
  993. }
  994. return 0;
  995. }
  996. void Set_VDI_Value(unsigned short axis,unsigned short vdi_value)
  997. {
  998. static unsigned long set_vdi_servo_x_delay=0;
  999. static unsigned long set_vdi_servo_y_delay=0;
  1000. static unsigned long set_vdi_servo_z_delay=0;
  1001. switch(axis)
  1002. {
  1003. case X_AXIS:
  1004. if(dwTickCount>=set_vdi_servo_x_delay)
  1005. {
  1006. set_vdi_servo_x_delay=dwTickCount+500;
  1007. servo_x.vdi_value=vdi_value;
  1008. servo_com_cmd(axis,VDI_VALUE_SET);
  1009. }
  1010. break;
  1011. case Y_AXIS:
  1012. if(dwTickCount>=set_vdi_servo_y_delay)
  1013. {
  1014. set_vdi_servo_y_delay=dwTickCount+500;
  1015. servo_y.vdi_value=vdi_value;
  1016. servo_com_cmd(axis,VDI_VALUE_SET);
  1017. }
  1018. break;
  1019. case Z_AXIS:
  1020. servo_z.vdi_value=vdi_value;
  1021. if(dwTickCount>=set_vdi_servo_z_delay)
  1022. {
  1023. set_vdi_servo_z_delay=dwTickCount+500;
  1024. servo_com_cmd(axis,VDI_VALUE_SET);
  1025. }
  1026. break;
  1027. }
  1028. }
  1029. /**
  1030. * 伺服通讯初初始化函数
  1031. *
  1032. * @author laz
  1033. *
  1034. * @param cmd
  1035. */
  1036. static int servo_com_init(unsigned short axis,servo_param_t *servo_axis)
  1037. {
  1038. //通讯握手三次
  1039. switch(servo_axis->init_step)
  1040. {
  1041. case 0://握手
  1042. servo_axis->IO_TO_COM=0;
  1043. servo_com_cmd(axis,HANDSHAKE);
  1044. //servo_axis->cmd_fb[HANDSHAKE]=0;//清除反馈命令
  1045. sw_timer_start(&servo_axis->timer, 1, 0);//1S超时
  1046. servo_axis->init_step++;
  1047. servo_axis->com_number++;
  1048. break;
  1049. case 1://判断握手情况。
  1050. if (sw_timer_expire(&servo_axis->timer)){//握手超时
  1051. if(servo_axis->com_number>=3){//三次通讯超时,通讯失败。
  1052. servo_axis->init_step=0xF0;
  1053. }
  1054. else{//没有到三次继续通讯。
  1055. servo_axis->init_step=0;
  1056. }
  1057. }
  1058. if(servo_axis->cmd_fb[HANDSHAKE]==HANDSHAKE)//通讯成功
  1059. {
  1060. servo_axis->com_ok_number++;
  1061. if(servo_axis->com_ok_number>=3){//通讯成功3次
  1062. servo_axis->init_step++;
  1063. servo_axis->com_number=0;
  1064. }
  1065. else{
  1066. servo_axis->init_step=0;
  1067. }
  1068. }
  1069. break;
  1070. case 2://配置伺服驱动器。
  1071. //停机关使能
  1072. //SetEn(axis, 1);
  1073. set_servo_mode(axis,RUN_MODE);
  1074. servo_axis->com_number++;
  1075. sw_timer_start(&servo_axis->timer, 2, 0);//1S超时
  1076. servo_axis->init_step++;
  1077. break;
  1078. case 3://判断配置IO情况。
  1079. if (sw_timer_expire(&servo_axis->timer)){//配置失败
  1080. if(servo_axis->com_number>=3){//三次配置失败,通讯失败。
  1081. servo_axis->init_step=0xF1;
  1082. }
  1083. else{//配置没有成功重新配置。
  1084. servo_axis->init_step=2;
  1085. }
  1086. }
  1087. //配置成功
  1088. if(servo_axis->cmd_fb[DI_MODE_SW]==DI_MODE_SW&&servo_axis->cmd_fb[DO_TARR_SW]==DO_TARR_SW
  1089. &&servo_axis->cmd_fb[CTRL_MODE_SET]==CTRL_MODE_SET
  1090. &&servo_axis->cmd_fb[TARR_SOURCE_SEL]==TARR_SOURCE_SEL
  1091. //&&(servo_axis->cmd_fb[SPEED_LIMIT_SET]==SPEED_LIMIT_SET&&RUN_MODE==0
  1092. //||servo_axis->cmd_fb[TARR_LIMIT12_SET]==TARR_LIMIT12_SET&&RUN_MODE==1)
  1093. &&servo_axis->cmd_fb[TARR_BASE_SET]==TARR_BASE_SET){
  1094. servo_axis->init_step++;
  1095. }
  1096. break;
  1097. case 4://初始化配置扭矩模式下的速度限制、扭矩限制、扭矩基准值
  1098. if(RUN_MODE==0)
  1099. {
  1100. set_servo_postotarr_limit(axis,550,10000,10000,SERVO_TARR_CCW);
  1101. }
  1102. else
  1103. {
  1104. set_servo_tarr_limit(axis,1000,1000,20000,10000,SERVO_TARR_CCW);
  1105. }
  1106. servo_axis->init_step++;
  1107. break;
  1108. case 5://设置告警和清除告警的读取模式,循环读取ALARMCODE;
  1109. Set_Servo_Runmode(axis,1);//设置伺服运行在通讯模式,
  1110. //servo_com_cmd(axis,ALARM_CODE);
  1111. Set_read_alarm(axis);
  1112. //servo_axis->cmd_fb[ALARM_CODE]=0;//清除反馈命令
  1113. servo_axis->init_step++;
  1114. break;
  1115. case 6://配置成功,输出使能
  1116. servo_axis->IO_TO_COM=1;//配置为IO模式
  1117. servo_axis->init_step=101;
  1118. //servo_com_cmd(axis,ENCODE_FB);
  1119. Set_read_encode(axis);
  1120. if(axis==Z_AXIS)
  1121. {
  1122. vdi_config_init(Z_AXIS);
  1123. //set_servo_tarr_limit(Z_AXIS,400,400,50000,50000,SERVO_TARR_CW);
  1124. }
  1125. return 1;
  1126. break;
  1127. case 0xF1://配置失败处理
  1128. servo_com_cmd(axis,SERVO_RESTART);//伺服重启
  1129. servo_axis->cmd_fb[SERVO_RESTART]=0;//清除反馈命令
  1130. servo_axis->init_step++;
  1131. break;
  1132. case 0xF2://配置失败处理
  1133. if(servo_axis->cmd_fb[SERVO_RESTART]==SERVO_RESTART){
  1134. ;//SetEn(X_AXIS, MOTOR_EN);//设置使能
  1135. return 0xF2;
  1136. }
  1137. case 0xF0://配置失败处理
  1138. return 0xF0;
  1139. break;
  1140. }
  1141. return 0;
  1142. }
  1143. //获取伺服通讯初始化状态
  1144. int GetServoComState(unsigned short axis)
  1145. {
  1146. switch(axis)
  1147. {
  1148. case X_AXIS:
  1149. return servo_x.IO_TO_COM;
  1150. break;
  1151. case Y_AXIS:
  1152. return servo_y.IO_TO_COM;
  1153. break;
  1154. case Z_AXIS:
  1155. return servo_z.IO_TO_COM;
  1156. break;
  1157. }
  1158. return 0;
  1159. }
  1160. //设置扭矩的应用
  1161. void SetServoComUse(unsigned short axis,unsigned short enable)
  1162. {
  1163. switch(axis)
  1164. {
  1165. case X_AXIS:
  1166. if(enable)
  1167. {
  1168. SetEn(X_AXIS, 0);
  1169. servo_x.init_step=0;
  1170. }
  1171. else
  1172. {
  1173. servo_x.init_step=0xF0;
  1174. }
  1175. break;
  1176. case Y_AXIS:
  1177. if(enable)
  1178. {
  1179. SetEn(Y_AXIS, 0);
  1180. servo_y.init_step=0;
  1181. }
  1182. else
  1183. {
  1184. servo_y.init_step=0xF0;
  1185. }
  1186. break;
  1187. case Z_AXIS:
  1188. if(enable)
  1189. {
  1190. SetEn(Z_AXIS, 0);
  1191. servo_z.init_step=0;
  1192. }
  1193. else
  1194. {
  1195. servo_z.init_step=0xF0;
  1196. }
  1197. break;
  1198. }
  1199. }
  1200. //伺服运行测试位置
  1201. static sw_timer_t servo_com_timer;//伺服通讯定时器。
  1202. unsigned char servo_test_step=0;
  1203. long X_pos;
  1204. int servo_test_pluse_mode(void)
  1205. {
  1206. static long servo_test_delay=0;
  1207. memcpy(&user_datas[610],&dwYRealPos,4);
  1208. memcpy(&user_datas[612],&olddwYRealPos[0],4);
  1209. SERVO_STEP=servo_test_step;
  1210. dwXRealPos = GetPos(X_AXIS);
  1211. switch(servo_test_step)
  1212. {
  1213. case 1:
  1214. SetPos(X_AXIS,0);
  1215. servo_test_step++;
  1216. case 2:
  1217. AxisMovePosAccDecNotStop(X_AXIS,10000,50000,1000,10000,20,80,50);
  1218. servo_test_step++;
  1219. break;
  1220. case 3://位置运行
  1221. if(abs(dwXRealPos)>=20000)
  1222. {
  1223. //Set_Ctrlmode_trans(X_AXIS,1);
  1224. servo_test_delay=dwTickCount + 1; //维持1S
  1225. servo_test_step++;
  1226. }
  1227. break;
  1228. case 4:
  1229. if(dwTickCount>=servo_test_delay)
  1230. {
  1231. AxisMovePosAccDec(X_AXIS,10000,2000,1000,1000,20,100,50);
  1232. Set_Ctrlmode_trans(X_AXIS,1);
  1233. servo_test_delay=dwTickCount + 1; //维持1S
  1234. servo_test_step++;
  1235. }
  1236. break;
  1237. case 5://
  1238. if(dwTickCount>=servo_test_delay)
  1239. Set_Ctrlmode_trans(X_AXIS,0);
  1240. if(!X_DRV){
  1241. servo_test_step=0;
  1242. //Set_Ctrlmode_trans(X_AXIS,0);
  1243. }
  1244. break;
  1245. case 8://松轴
  1246. AxisMovePosAccDecNotStop(Y_AXIS,2000,-50000,1000,1000,20,50,50);
  1247. servo_test_step++;
  1248. break;
  1249. case 9:
  1250. if(get_tarr_set(Y_AXIS)==2)//设置扭矩完成
  1251. {
  1252. if(abs(dwYRealPos)>=10000)
  1253. {
  1254. Set_Ctrlmode_trans(Y_AXIS,TARR_MODE);//设置成为扭矩模式
  1255. servo_test_delay=dwTickCount + 5000;
  1256. servo_test_step++;
  1257. }
  1258. }
  1259. else if(get_tarr_set(Y_AXIS)==0)//配置超时
  1260. {
  1261. servo_test_step=0xF0;
  1262. }
  1263. break;
  1264. case 10://
  1265. if(dwTickCount>=servo_test_delay)
  1266. {
  1267. if(GetTarr(Y_AXIS))//转矩到达
  1268. {
  1269. servo_test_delay=dwTickCount + 1000; //维持1S
  1270. servo_test_step++;
  1271. }
  1272. }
  1273. break;
  1274. case 11://
  1275. if(dwTickCount>=servo_test_delay)
  1276. {
  1277. if(GetTarr(Y_AXIS))
  1278. {
  1279. Set_Ctrlmode_trans(Y_AXIS,POS_MODE);//转位置模式;
  1280. servo_test_delay=dwTickCount + 100;
  1281. servo_test_step++;
  1282. }
  1283. }
  1284. break;
  1285. case 12://1S后转转矩模式
  1286. if(dwTickCount>=servo_test_delay)
  1287. {
  1288. Set_Ctrlmode_trans(Y_AXIS,TARR_MODE);//转位置模式;
  1289. servo_test_step++;
  1290. }
  1291. break;
  1292. case 13://
  1293. if(GetTarr(Y_AXIS))//转矩到达
  1294. {
  1295. servo_test_delay=dwTickCount + 1000; //维持1S
  1296. AxisEgmStop(Y_AXIS);
  1297. servo_test_step++;
  1298. }
  1299. break;
  1300. case 14://
  1301. if(dwTickCount>=servo_test_delay)
  1302. {
  1303. if(GetTarr(Y_AXIS))
  1304. {
  1305. Set_Ctrlmode_trans(Y_AXIS,POS_MODE);//转位置模式;
  1306. servo_test_step=0;
  1307. }
  1308. }
  1309. break;
  1310. case 20://配置扭矩
  1311. //SetEn(X_AXIS, MOTOR_EN);
  1312. //SetPos(X_AXIS,0);
  1313. SetPos(Y_AXIS,0);
  1314. //扭矩模式,用扭矩1和扭矩2之间的切换
  1315. //扭矩模式+位置模式(轴--最大扭矩--最小扭矩--正转速度限制--反转速度限制--方向)
  1316. //set_servo_tarr_limit(X_AXIS,200,100,400,100,SERVO_TARR_CW);
  1317. set_servo_tarr_limit(Y_AXIS,200,100,4000,2000,SERVO_TARR_CW);
  1318. //设置最大扭矩限制
  1319. Set_Ctrlmode_trans(Y_AXIS,TARR_MODE_MAXLIMIT);
  1320. //
  1321. //Set_Ctrlmode_trans(Y_AXIS,TARR_MODE_LIMIT1);
  1322. sw_timer_start(&servo_com_timer, 10, 0);
  1323. //servo_test_step++;
  1324. break;
  1325. case 21://扭矩运行
  1326. if(get_tarr_set(Y_AXIS)==2)//设置扭矩完成
  1327. {
  1328. if(abs(dwYRealPos)>=10000)
  1329. {
  1330. SetEn(Y_AXIS, MOTOR_EN);//运行
  1331. servo_test_step++;
  1332. }
  1333. }
  1334. else if(get_tarr_set(Y_AXIS)==0)//配置超时
  1335. {
  1336. servo_test_step=0xF0;
  1337. }
  1338. break;
  1339. case 22://维持运行
  1340. servo_test_delay=dwTickCount + 10000; //维持10S
  1341. servo_test_step++;
  1342. break;
  1343. case 23://切换扭矩,并维持运行
  1344. if(dwTickCount>=servo_test_delay)
  1345. {
  1346. Set_Ctrlmode_trans(Y_AXIS,TARR_MODE_MINLIMIT); //切换成扭矩
  1347. servo_test_delay=dwTickCount + 10000; //维持10S
  1348. servo_test_step++;
  1349. }
  1350. break;
  1351. case 24://停止,切换方向,重新设置
  1352. if(dwTickCount>=servo_test_delay)
  1353. {
  1354. SetEn(Y_AXIS, MOTOR_EN);//停止
  1355. set_servo_tarr_limit(Y_AXIS,200,100,4000,2000,SERVO_TARR_CCW);//切换方向
  1356. Set_Ctrlmode_trans(Y_AXIS,TARR_MODE_MAXLIMIT);
  1357. servo_test_step++;
  1358. }
  1359. break;
  1360. case 25://运行
  1361. if(get_tarr_set(Y_AXIS)==2)//设置扭矩完成
  1362. {
  1363. if(abs(dwYRealPos)>=10000)
  1364. {
  1365. SetEn(Y_AXIS, MOTOR_EN);//运行
  1366. servo_test_step++;
  1367. }
  1368. }
  1369. else if(get_tarr_set(Y_AXIS)==0)//配置超时
  1370. {
  1371. servo_test_step=0xF0;
  1372. }
  1373. break;
  1374. case 26://维持运行
  1375. servo_test_delay=dwTickCount + 10000; //维持10S
  1376. servo_test_step++;
  1377. break;
  1378. case 27://切换扭矩
  1379. if(dwTickCount>=servo_test_delay)
  1380. {
  1381. Set_Ctrlmode_trans(Y_AXIS,TARR_MODE_MINLIMIT); //切换成扭矩
  1382. servo_test_delay=dwTickCount + 10000; //维持10S
  1383. servo_test_step++;
  1384. }
  1385. break;
  1386. case 28://停止
  1387. if(dwTickCount>=servo_test_delay)
  1388. {
  1389. SetEn(Y_AXIS, MOTOR_EN);//停止
  1390. servo_test_step=0;
  1391. }
  1392. break;
  1393. }
  1394. if(M0045)
  1395. {
  1396. M0045=0;
  1397. AxisContinueMoveChangeSpeed(Y_AXIS,CHANGE_SPEED,STOP_SPEED,10,10);
  1398. }
  1399. if(M0046)
  1400. {
  1401. M0046=0;
  1402. servo_test_step++;
  1403. }
  1404. if(M0047)
  1405. {
  1406. M0047=0;
  1407. //配置扭矩模式下的速度限制。
  1408. servo_y.speed_plimit_set=LIMIT_SPEED;
  1409. servo_y.speed_nlimit_set=LIMIT_SPEED;
  1410. servo_com_cmd(Y_AXIS,SPEED_LIMIT_SET);
  1411. servo_y.cmd_fb[SPEED_LIMIT_SET]=0;//清除反馈命令
  1412. }
  1413. return 0;
  1414. }
  1415. //伺服运行
  1416. int servo_com_run(void)
  1417. {
  1418. int runflag;
  1419. #if X_USERING_TARR==1
  1420. runflag=servo_com_init(X_AXIS,&servo_x);//初始化
  1421. servo_com_get_alarm(X_AXIS);
  1422. #endif
  1423. #if Y_USERING_TARR==1
  1424. runflag=servo_com_init(Y_AXIS,&servo_y);//初始化
  1425. servo_com_get_alarm(Y_AXIS);
  1426. #endif
  1427. #if Z_USERING_TARR==1
  1428. runflag=servo_com_init(Z_AXIS,&servo_z);//初始化
  1429. servo_com_get_alarm(Z_AXIS);
  1430. #endif
  1431. servo_test_pluse_mode();
  1432. return runflag;
  1433. }