/******************************************************************** * Description: shcom.cc * Common functions for NML calls * * Derived from a work by Fred Proctor & Will Shackleford * Further derived from work by jmkasunich, Alex Joni * * Author: Eric H. Johnson * License: GPL Version 2 * System: Linux * * Copyright (c) 2006 All rights reserved. * * Last change: ********************************************************************/ #define __STDC_FORMAT_MACROS #include #include #include #include #include #include #include #include #include #include "emc/linuxcnc.h" #include "rcs.hh" #include "posemath.h" // PM_POSE, TO_RAD #include "emc.hh" // EMC NML #include "emc_nml.hh" #include "canon.hh" // CANON_UNITS, CANON_UNITS_INCHES,MM,CM #include "emcglb.h" // EMC_NMLFILE, TRAJ_MAX_VELOCITY, etc. #include "emccfg.h" // DEFAULT_TRAJ_MAX_VELOCITY #include "inifile.hh" // INIFILE #include "nml_oi.hh" // nmlErrorFormat, NML_ERROR, etc #include "rcs_print.hh" #include "timer.hh" // esleep #include "shcom.hh" // Common NML communications functions #include LINEAR_UNIT_CONVERSION linearUnitConversion; ANGULAR_UNIT_CONVERSION angularUnitConversion; static int num_joints = EMCMOT_MAX_JOINTS; int emcCommandSerialNumber; // the NML channels to the EMC task RCS_CMD_CHANNEL *emcCommandBuffer; RCS_STAT_CHANNEL *emcStatusBuffer; EMC_STAT *emcStatus; // the NML channel for errors NML *emcErrorBuffer; char error_string[NML_ERROR_LEN]; char operator_text_string[NML_TEXT_LEN]; char operator_display_string[NML_DISPLAY_LEN]; char defaultPath[80] = DEFAULT_PATH; // default value for timeout, 0 means wait forever double emcTimeout; int programStartLine; EMC_UPDATE_TYPE emcUpdateType; EMC_WAIT_TYPE emcWaitType; void strupr(char *s) { int i; for (i = 0; i < (int)strlen(s); i++) if (s[i] > 96 && s[i] <= 'z') s[i] -= 32; } int emcTaskNmlGet() { int retval = 0; // try to connect to EMC cmd if (emcCommandBuffer == 0) { emcCommandBuffer = new RCS_CMD_CHANNEL(emcFormat, "emcCommand", "xemc", emc_nmlfile); if (!emcCommandBuffer->valid()) { delete emcCommandBuffer; emcCommandBuffer = 0; retval = -1; } } // try to connect to EMC status if (emcStatusBuffer == 0) { emcStatusBuffer = new RCS_STAT_CHANNEL(emcFormat, "emcStatus", "xemc", emc_nmlfile); if (!emcStatusBuffer->valid() || EMC_STAT_TYPE != emcStatusBuffer->peek()) { delete emcStatusBuffer; emcStatusBuffer = 0; emcStatus = 0; retval = -1; } else { emcStatus = (EMC_STAT *) emcStatusBuffer->get_address(); } } return retval; } int emcErrorNmlGet() { int retval = 0; if (emcErrorBuffer == 0) { emcErrorBuffer = new NML(nmlErrorFormat, "emcError", "xemc", emc_nmlfile); if (!emcErrorBuffer->valid()) { delete emcErrorBuffer; emcErrorBuffer = 0; retval = -1; } } return retval; } int tryNml(double retry_time, double retry_interval) { double end; int good; if ((emc_debug & EMC_DEBUG_NML) == 0) { set_rcs_print_destination(RCS_PRINT_TO_NULL); // inhibit diag // messages } end = retry_time; good = 0; do { if (0 == emcTaskNmlGet()) { good = 1; break; } esleep(retry_interval); end -= retry_interval; } while (end > 0.0); if ((emc_debug & EMC_DEBUG_NML) == 0) { set_rcs_print_destination(RCS_PRINT_TO_STDOUT); // inhibit diag // messages } if (!good) { return -1; } if ((emc_debug & EMC_DEBUG_NML) == 0) { set_rcs_print_destination(RCS_PRINT_TO_NULL); // inhibit diag // messages } end = retry_time; good = 0; do { if (0 == emcErrorNmlGet()) { good = 1; break; } esleep(retry_interval); end -= retry_interval; } while (end > 0.0); if ((emc_debug & EMC_DEBUG_NML) == 0) { set_rcs_print_destination(RCS_PRINT_TO_STDOUT); // inhibit diag // messages } if (!good) { return -1; } return 0; } int updateStatus() { NMLTYPE type; if (0 == emcStatus || 0 == emcStatusBuffer || !emcStatusBuffer->valid()) { return -1; } switch (type = emcStatusBuffer->peek()) { case -1: // error on CMS channel return -1; break; case 0: // no new data case EMC_STAT_TYPE: // new data // new data break; default: return -1; break; } return 0; } /* updateError() updates "errors," which are true errors and also operator display and text messages. */ int updateError() { NMLTYPE type; if (0 == emcErrorBuffer || !emcErrorBuffer->valid()) { return -1; } switch (type = emcErrorBuffer->read()) { case -1: // error reading channel return -1; break; case 0: // nothing new break; case EMC_OPERATOR_ERROR_TYPE: strncpy(error_string, ((EMC_OPERATOR_ERROR *) (emcErrorBuffer->get_address()))-> error, LINELEN - 1); error_string[NML_ERROR_LEN - 1] = 0; break; case EMC_OPERATOR_TEXT_TYPE: strncpy(operator_text_string, ((EMC_OPERATOR_TEXT *) (emcErrorBuffer->get_address()))-> text, LINELEN - 1); operator_text_string[NML_TEXT_LEN - 1] = 0; break; case EMC_OPERATOR_DISPLAY_TYPE: strncpy(operator_display_string, ((EMC_OPERATOR_DISPLAY *) (emcErrorBuffer-> get_address()))->display, LINELEN - 1); operator_display_string[NML_DISPLAY_LEN - 1] = 0; break; case NML_ERROR_TYPE: strncpy(error_string, ((NML_ERROR *) (emcErrorBuffer->get_address()))->error, NML_ERROR_LEN - 1); error_string[NML_ERROR_LEN - 1] = 0; break; case NML_TEXT_TYPE: strncpy(operator_text_string, ((NML_TEXT *) (emcErrorBuffer->get_address()))->text, NML_TEXT_LEN - 1); operator_text_string[NML_TEXT_LEN - 1] = 0; break; case NML_DISPLAY_TYPE: strncpy(operator_display_string, ((NML_DISPLAY *) (emcErrorBuffer->get_address()))->display, NML_DISPLAY_LEN - 1); operator_display_string[NML_DISPLAY_LEN - 1] = 0; break; default: // if not recognized, set the error string snprintf(error_string, sizeof(error_string), "unrecognized error %" PRId32, type); return -1; break; } return 0; } #define EMC_COMMAND_DELAY 0.1 // how long to sleep between checks int emcCommandWaitDone() { double end; for (end = 0.0; emcTimeout <= 0.0 || end < emcTimeout; end += EMC_COMMAND_DELAY) { updateStatus(); int serial_diff = emcStatus->echo_serial_number - emcCommandSerialNumber; if (serial_diff < 0) { continue; } if (serial_diff > 0) { return 0; } if (emcStatus->status == RCS_DONE) { return 0; } if (emcStatus->status == RCS_ERROR) { return -1; } esleep(EMC_COMMAND_DELAY); } return -1; } int emcCommandWaitReceived() { double end; for (end = 0.0; emcTimeout <= 0.0 || end < emcTimeout; end += EMC_COMMAND_DELAY) { updateStatus(); int serial_diff = emcStatus->echo_serial_number - emcCommandSerialNumber; if (serial_diff >= 0) { return 0; } esleep(EMC_COMMAND_DELAY); } return -1; } int emcCommandSend(RCS_CMD_MSG & cmd) { // write command if (emcCommandBuffer->write(&cmd)) { return -1; } emcCommandSerialNumber = cmd.serial_number; return 0; } /* Unit conversion Length and angle units in the EMC status buffer are in user units, as defined in the INI file in [TRAJ] LINEAR,ANGULAR_UNITS. These may differ from the program units, and when they are the display is confusing. It may be desirable to synchronize the display units with the program units automatically, and also to break this sync and allow independent display of position values. The global variable "linearUnitConversion" is set by the Tcl commands emc_linear_unit_conversion to correspond to either "inch", "mm", "cm", "auto", or "custom". This forces numbers to be returned in the units specified, in program units when "auto" is set, or not converted at all if "custom" is specified. Ditto for "angularUnitConversion", set by emc_angular_unit_conversion to "deg", "rad", "grad", "auto", or "custom". With no args, emc_linear/angular_unit_conversion return the setting. The functions convertLinearUnits and convertAngularUnits take a length or angle value, typically from the emcStatus structure, and convert it as indicated by linearUnitConversion and angularUnitConversion, resp. */ /* to convert linear units, values are converted to mm, then to desired units */ double convertLinearUnits(double u) { double in_mm; /* convert u to mm */ in_mm = u / emcStatus->motion.traj.linearUnits; /* convert u to display units */ switch (linearUnitConversion) { case LINEAR_UNITS_MM: return in_mm; break; case LINEAR_UNITS_INCH: return in_mm * INCH_PER_MM; break; case LINEAR_UNITS_CM: return in_mm * CM_PER_MM; break; case LINEAR_UNITS_AUTO: switch (emcStatus->task.programUnits) { case CANON_UNITS_MM: return in_mm; break; case CANON_UNITS_INCHES: return in_mm * INCH_PER_MM; break; case CANON_UNITS_CM: return in_mm * CM_PER_MM; break; } break; case LINEAR_UNITS_CUSTOM: return u; break; } // If it ever gets here we have an error. return u; } double convertAngularUnits(double u) { // Angular units are always degrees return u; } // polarities for joint jogging, from INI file static int jogPol[EMCMOT_MAX_JOINTS]; int sendDebug(int level) { EMC_SET_DEBUG debug_msg; debug_msg.debug = level; emcCommandSend(debug_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendEstop() { EMC_TASK_SET_STATE state_msg; state_msg.state = EMC_TASK_STATE_ESTOP; emcCommandSend(state_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendEstopReset() { EMC_TASK_SET_STATE state_msg; state_msg.state = EMC_TASK_STATE_ESTOP_RESET; emcCommandSend(state_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendMachineOn() { EMC_TASK_SET_STATE state_msg; state_msg.state = EMC_TASK_STATE_ON; emcCommandSend(state_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendMachineOff() { EMC_TASK_SET_STATE state_msg; state_msg.state = EMC_TASK_STATE_OFF; emcCommandSend(state_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendManual() { EMC_TASK_SET_MODE mode_msg; mode_msg.mode = EMC_TASK_MODE_MANUAL; emcCommandSend(mode_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendAuto() { EMC_TASK_SET_MODE mode_msg; mode_msg.mode = EMC_TASK_MODE_AUTO; emcCommandSend(mode_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendMdi() { EMC_TASK_SET_MODE mode_msg; mode_msg.mode = EMC_TASK_MODE_MDI; emcCommandSend(mode_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendOverrideLimits(int joint) { EMC_JOINT_OVERRIDE_LIMITS lim_msg; lim_msg.joint = joint; // neg means off, else on for all emcCommandSend(lim_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendJogStop(int ja, int jjogmode) { EMC_JOG_STOP emc_jog_stop_msg; if ( ( (jjogmode == JOGJOINT) && (emcStatus->motion.traj.mode == EMC_TRAJ_MODE_TELEOP) ) || ( (jjogmode == JOGTELEOP ) && (emcStatus->motion.traj.mode != EMC_TRAJ_MODE_TELEOP) ) ) { return -1; } if ( jjogmode && (ja < 0 || ja >= num_joints)) { fprintf(stderr,"shcom.cc: unexpected_1 %d\n",ja); return -1; } if ( !jjogmode && (ja < 0)) { fprintf(stderr,"shcom.cc: unexpected_2 %d\n",ja); return -1; } emc_jog_stop_msg.jjogmode = jjogmode; emc_jog_stop_msg.joint_or_axis = ja; emcCommandSend(emc_jog_stop_msg); return 0; } int sendJogCont(int ja, int jjogmode, double speed) { EMC_JOG_CONT emc_jog_cont_msg; if (emcStatus->task.state != EMC_TASK_STATE_ON) { return -1; } if ( ( (jjogmode == JOGJOINT) && (emcStatus->motion.traj.mode == EMC_TRAJ_MODE_TELEOP) ) || ( (jjogmode == JOGTELEOP ) && (emcStatus->motion.traj.mode != EMC_TRAJ_MODE_TELEOP) ) ) { return -1; } if ( jjogmode && (ja < 0 || ja >= num_joints)) { fprintf(stderr,"shcom.cc: unexpected_3 %d\n",ja); return -1; } if ( !jjogmode && (ja < 0)) { fprintf(stderr,"shcom.cc: unexpected_4 %d\n",ja); return -1; } emc_jog_cont_msg.jjogmode = jjogmode; emc_jog_cont_msg.joint_or_axis = ja; emc_jog_cont_msg.vel = speed / 60.0; emcCommandSend(emc_jog_cont_msg); return 0; } int sendJogIncr(int ja, int jjogmode, double speed, double incr) { EMC_JOG_INCR emc_jog_incr_msg; if (emcStatus->task.state != EMC_TASK_STATE_ON) { return -1; } if ( ( (jjogmode == JOGJOINT) && ( emcStatus->motion.traj.mode == EMC_TRAJ_MODE_TELEOP) ) || ( (jjogmode == JOGTELEOP ) && ( emcStatus->motion.traj.mode != EMC_TRAJ_MODE_TELEOP) ) ) { return -1; } if ( jjogmode && (ja < 0 || ja >= num_joints)) { fprintf(stderr,"shcom.cc: unexpected_5 %d\n",ja); return -1; } if ( !jjogmode && (ja < 0)) { fprintf(stderr,"shcom.cc: unexpected_6 %d\n",ja); return -1; } emc_jog_incr_msg.jjogmode = jjogmode; emc_jog_incr_msg.joint_or_axis = ja; emc_jog_incr_msg.vel = speed / 60.0; emc_jog_incr_msg.incr = incr; emcCommandSend(emc_jog_incr_msg); return 0; } int sendMistOn() { EMC_COOLANT_MIST_ON emc_coolant_mist_on_msg; emcCommandSend(emc_coolant_mist_on_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendMistOff() { EMC_COOLANT_MIST_OFF emc_coolant_mist_off_msg; emcCommandSend(emc_coolant_mist_off_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendFloodOn() { EMC_COOLANT_FLOOD_ON emc_coolant_flood_on_msg; emcCommandSend(emc_coolant_flood_on_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendFloodOff() { EMC_COOLANT_FLOOD_OFF emc_coolant_flood_off_msg; emcCommandSend(emc_coolant_flood_off_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendLubeOn() { EMC_LUBE_ON emc_lube_on_msg; emcCommandSend(emc_lube_on_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendLubeOff() { EMC_LUBE_OFF emc_lube_off_msg; emcCommandSend(emc_lube_off_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendSpindleForward(int spindle) { EMC_SPINDLE_ON emc_spindle_on_msg; emc_spindle_on_msg.spindle = spindle; if (emcStatus->task.activeSettings[2] != 0) { emc_spindle_on_msg.speed = fabs(emcStatus->task.activeSettings[2]); } else { emc_spindle_on_msg.speed = +500; } emcCommandSend(emc_spindle_on_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendSpindleReverse(int spindle) { EMC_SPINDLE_ON emc_spindle_on_msg; emc_spindle_on_msg.spindle = spindle; if (emcStatus->task.activeSettings[2] != 0) { emc_spindle_on_msg.speed = -1 * fabs(emcStatus->task.activeSettings[2]); } else { emc_spindle_on_msg.speed = -500; } emcCommandSend(emc_spindle_on_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendSpindleOff(int spindle) { EMC_SPINDLE_OFF emc_spindle_off_msg; emc_spindle_off_msg.spindle = spindle; emcCommandSend(emc_spindle_off_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendSpindleIncrease(int spindle) { EMC_SPINDLE_INCREASE emc_spindle_increase_msg; emc_spindle_increase_msg.spindle = spindle; emcCommandSend(emc_spindle_increase_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendSpindleDecrease(int spindle) { EMC_SPINDLE_DECREASE emc_spindle_decrease_msg; emc_spindle_decrease_msg.spindle = spindle; emcCommandSend(emc_spindle_decrease_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendSpindleConstant(int spindle) { EMC_SPINDLE_CONSTANT emc_spindle_constant_msg; emc_spindle_constant_msg.spindle = spindle; emcCommandSend(emc_spindle_constant_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendBrakeEngage(int spindle) { EMC_SPINDLE_BRAKE_ENGAGE emc_spindle_brake_engage_msg; emc_spindle_brake_engage_msg.spindle = spindle; emcCommandSend(emc_spindle_brake_engage_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendBrakeRelease(int spindle) { EMC_SPINDLE_BRAKE_RELEASE emc_spindle_brake_release_msg; emc_spindle_brake_release_msg.spindle = spindle; emcCommandSend(emc_spindle_brake_release_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendAbort() { EMC_TASK_ABORT task_abort_msg; emcCommandSend(task_abort_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendHome(int joint) { EMC_JOINT_HOME emc_joint_home_msg; emc_joint_home_msg.joint = joint; emcCommandSend(emc_joint_home_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendUnHome(int joint) { EMC_JOINT_UNHOME emc_joint_home_msg; emc_joint_home_msg.joint = joint; emcCommandSend(emc_joint_home_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendFeedOverride(double override) { EMC_TRAJ_SET_SCALE emc_traj_set_scale_msg; if (override < 0.0) { override = 0.0; } emc_traj_set_scale_msg.scale = override; emcCommandSend(emc_traj_set_scale_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendRapidOverride(double override) { EMC_TRAJ_SET_RAPID_SCALE emc_traj_set_scale_msg; if (override < 0.0) { override = 0.0; } if (override > 1.0) { override = 1.0; } emc_traj_set_scale_msg.scale = override; emcCommandSend(emc_traj_set_scale_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendSpindleOverride(int spindle, double override) { EMC_TRAJ_SET_SPINDLE_SCALE emc_traj_set_spindle_scale_msg; if (override < 0.0) { override = 0.0; } emc_traj_set_spindle_scale_msg.spindle = spindle; emc_traj_set_spindle_scale_msg.scale = override; emcCommandSend(emc_traj_set_spindle_scale_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendTaskPlanInit() { EMC_TASK_PLAN_INIT task_plan_init_msg; emcCommandSend(task_plan_init_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } // saved value of last program opened static char lastProgramFile[LINELEN] = ""; int sendProgramOpen(char *program) { EMC_TASK_PLAN_OPEN emc_task_plan_open_msg; // save this to run again rtapi_strxcpy(lastProgramFile, program); rtapi_strxcpy(emc_task_plan_open_msg.file, program); emcCommandSend(emc_task_plan_open_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendProgramRun(int line) { EMC_TASK_PLAN_RUN emc_task_plan_run_msg; if (emcUpdateType == EMC_UPDATE_AUTO) { updateStatus(); } // first reopen program if it's not open if (0 == emcStatus->task.file[0]) { // send a request to open last one sendProgramOpen(lastProgramFile); } // save the start line, to compare against active line later programStartLine = line; emc_task_plan_run_msg.line = line; emcCommandSend(emc_task_plan_run_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendProgramPause() { EMC_TASK_PLAN_PAUSE emc_task_plan_pause_msg; emcCommandSend(emc_task_plan_pause_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendProgramResume() { EMC_TASK_PLAN_RESUME emc_task_plan_resume_msg; emcCommandSend(emc_task_plan_resume_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendSetOptionalStop(bool state) { EMC_TASK_PLAN_SET_OPTIONAL_STOP emc_task_plan_set_optional_stop_msg; emc_task_plan_set_optional_stop_msg.state = state; emcCommandSend(emc_task_plan_set_optional_stop_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendProgramStep() { EMC_TASK_PLAN_STEP emc_task_plan_step_msg; // clear out start line, if we had a verify before it would be -1 programStartLine = 0; emcCommandSend(emc_task_plan_step_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendMdiCmd(const char *mdi) { EMC_TASK_PLAN_EXECUTE emc_task_plan_execute_msg; rtapi_strxcpy(emc_task_plan_execute_msg.command, mdi); emcCommandSend(emc_task_plan_execute_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendLoadToolTable(const char *file) { EMC_TOOL_LOAD_TOOL_TABLE emc_tool_load_tool_table_msg; rtapi_strxcpy(emc_tool_load_tool_table_msg.file, file); emcCommandSend(emc_tool_load_tool_table_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendToolSetOffset(int toolno, double zoffset, double diameter) { EMC_TOOL_SET_OFFSET emc_tool_set_offset_msg; emc_tool_set_offset_msg.toolno = toolno; emc_tool_set_offset_msg.offset.tran.z = zoffset; emc_tool_set_offset_msg.diameter = diameter; emc_tool_set_offset_msg.orientation = 0; // mill style tool table emcCommandSend(emc_tool_set_offset_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendToolSetOffset(int toolno, double zoffset, double xoffset, double diameter, double frontangle, double backangle, int orientation) { EMC_TOOL_SET_OFFSET emc_tool_set_offset_msg; emc_tool_set_offset_msg.toolno = toolno; emc_tool_set_offset_msg.offset.tran.z = zoffset; emc_tool_set_offset_msg.offset.tran.x = xoffset; emc_tool_set_offset_msg.diameter = diameter; emc_tool_set_offset_msg.frontangle = frontangle; emc_tool_set_offset_msg.backangle = backangle; emc_tool_set_offset_msg.orientation = orientation; emcCommandSend(emc_tool_set_offset_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendJointSetBacklash(int joint, double backlash) { EMC_JOINT_SET_BACKLASH emc_joint_set_backlash_msg; emc_joint_set_backlash_msg.joint = joint; emc_joint_set_backlash_msg.backlash = backlash; emcCommandSend(emc_joint_set_backlash_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendJointEnable(int joint, int val) { EMC_JOINT_ENABLE emc_joint_enable_msg; EMC_JOINT_DISABLE emc_joint_disable_msg; if (val) { emc_joint_enable_msg.joint = joint; emcCommandSend(emc_joint_enable_msg); } else { emc_joint_disable_msg.joint = joint; emcCommandSend(emc_joint_disable_msg); } if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendJointLoadComp(int joint, const char *file, int type) { EMC_JOINT_LOAD_COMP emc_joint_load_comp_msg; rtapi_strxcpy(emc_joint_load_comp_msg.file, file); emc_joint_load_comp_msg.type = type; emcCommandSend(emc_joint_load_comp_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendSetTeleopEnable(int enable) { EMC_TRAJ_SET_TELEOP_ENABLE emc_set_teleop_enable_msg; emc_set_teleop_enable_msg.enable = enable; emcCommandSend(emc_set_teleop_enable_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendClearProbeTrippedFlag() { EMC_TRAJ_CLEAR_PROBE_TRIPPED_FLAG emc_clear_probe_tripped_flag_msg; emc_clear_probe_tripped_flag_msg.serial_number = emcCommandSend(emc_clear_probe_tripped_flag_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int sendProbe(double x, double y, double z) { EMC_TRAJ_PROBE emc_probe_msg; emc_probe_msg.pos.tran.x = x; emc_probe_msg.pos.tran.y = y; emc_probe_msg.pos.tran.z = z; emcCommandSend(emc_probe_msg); if (emcWaitType == EMC_WAIT_RECEIVED) { return emcCommandWaitReceived(); } else if (emcWaitType == EMC_WAIT_DONE) { return emcCommandWaitDone(); } return 0; } int iniLoad(const char *filename) { IniFile inifile; const char *inistring; char displayString[LINELEN] = ""; int t; int i; // open it if (inifile.Open(filename) == false) { return -1; } if (NULL != (inistring = inifile.Find("DEBUG", "EMC"))) { // copy to global if (1 != sscanf(inistring, "%i", &emc_debug)) { emc_debug = 0; } } else { // not found, use default emc_debug = 0; } if (NULL != (inistring = inifile.Find("NML_FILE", "EMC"))) { // copy to global rtapi_strxcpy(emc_nmlfile, inistring); } else { // not found, use default } for (t = 0; t < EMCMOT_MAX_JOINTS; t++) { jogPol[t] = 1; // set to default snprintf(displayString, sizeof(displayString), "JOINT_%d", t); if (NULL != (inistring = inifile.Find("JOGGING_POLARITY", displayString)) && 1 == sscanf(inistring, "%d", &i) && i == 0) { // it read as 0, so override default jogPol[t] = 0; } } if (NULL != (inistring = inifile.Find("LINEAR_UNITS", "DISPLAY"))) { if (!strcmp(inistring, "AUTO")) { linearUnitConversion = LINEAR_UNITS_AUTO; } else if (!strcmp(inistring, "INCH")) { linearUnitConversion = LINEAR_UNITS_INCH; } else if (!strcmp(inistring, "MM")) { linearUnitConversion = LINEAR_UNITS_MM; } else if (!strcmp(inistring, "CM")) { linearUnitConversion = LINEAR_UNITS_CM; } } else { // not found, leave default alone } if (NULL != (inistring = inifile.Find("ANGULAR_UNITS", "DISPLAY"))) { if (!strcmp(inistring, "AUTO")) { angularUnitConversion = ANGULAR_UNITS_AUTO; } else if (!strcmp(inistring, "DEG")) { angularUnitConversion = ANGULAR_UNITS_DEG; } else if (!strcmp(inistring, "RAD")) { angularUnitConversion = ANGULAR_UNITS_RAD; } else if (!strcmp(inistring, "GRAD")) { angularUnitConversion = ANGULAR_UNITS_GRAD; } } else { // not found, leave default alone } // close it inifile.Close(); return 0; } int checkStatus () { if (emcStatus) return 1; return 0; }