servocom_app.c 44 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441
  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. }