diff options
Diffstat (limited to 'ArduinoAddons/Arduino_1.6.x/libraries/TMC26XStepper/TMC26XStepper.cpp')
-rwxr-xr-x | ArduinoAddons/Arduino_1.6.x/libraries/TMC26XStepper/TMC26XStepper.cpp | 999 |
1 files changed, 999 insertions, 0 deletions
diff --git a/ArduinoAddons/Arduino_1.6.x/libraries/TMC26XStepper/TMC26XStepper.cpp b/ArduinoAddons/Arduino_1.6.x/libraries/TMC26XStepper/TMC26XStepper.cpp new file mode 100755 index 000000000..389264ca4 --- /dev/null +++ b/ArduinoAddons/Arduino_1.6.x/libraries/TMC26XStepper/TMC26XStepper.cpp @@ -0,0 +1,999 @@ +/* + TMC26XStepper.cpp - - TMC26X Stepper library for Wiring/Arduino + + based on the stepper library by Tom Igoe, et. al. + + Copyright (c) 2011, Interactive Matter, Marcus Nowotny + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in + all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + THE SOFTWARE. + + */ + +#if defined(ARDUINO) && ARDUINO >= 100 + #include <Arduino.h> +#else + #include <WProgram.h> +#endif +#include <SPI.h> +#include "TMC26XStepper.h" + +//some default values used in initialization +#define DEFAULT_MICROSTEPPING_VALUE 32 + +//TMC26X register definitions +#define DRIVER_CONTROL_REGISTER 0x0ul +#define CHOPPER_CONFIG_REGISTER 0x80000ul +#define COOL_STEP_REGISTER 0xA0000ul +#define STALL_GUARD2_LOAD_MEASURE_REGISTER 0xC0000ul +#define DRIVER_CONFIG_REGISTER 0xE0000ul + +#define REGISTER_BIT_PATTERN 0xFFFFFul + +//definitions for the driver control register +#define MICROSTEPPING_PATTERN 0xFul +#define STEP_INTERPOLATION 0x200ul +#define DOUBLE_EDGE_STEP 0x100ul +#define VSENSE 0x40ul +#define READ_MICROSTEP_POSTION 0x0ul +#define READ_STALL_GUARD_READING 0x10ul +#define READ_STALL_GUARD_AND_COOL_STEP 0x20ul +#define READ_SELECTION_PATTERN 0x30ul + +//definitions for the chopper config register +#define CHOPPER_MODE_STANDARD 0x0ul +#define CHOPPER_MODE_T_OFF_FAST_DECAY 0x4000ul +#define T_OFF_PATTERN 0xful +#define RANDOM_TOFF_TIME 0x2000ul +#define BLANK_TIMING_PATTERN 0x18000ul +#define BLANK_TIMING_SHIFT 15 +#define HYSTERESIS_DECREMENT_PATTERN 0x1800ul +#define HYSTERESIS_DECREMENT_SHIFT 11 +#define HYSTERESIS_LOW_VALUE_PATTERN 0x780ul +#define HYSTERESIS_LOW_SHIFT 7 +#define HYSTERESIS_START_VALUE_PATTERN 0x78ul +#define HYSTERESIS_START_VALUE_SHIFT 4 +#define T_OFF_TIMING_PATERN 0xFul + +//definitions for cool step register +#define MINIMUM_CURRENT_FOURTH 0x8000ul +#define CURRENT_DOWN_STEP_SPEED_PATTERN 0x6000ul +#define SE_MAX_PATTERN 0xF00ul +#define SE_CURRENT_STEP_WIDTH_PATTERN 0x60ul +#define SE_MIN_PATTERN 0xful + +//definitions for stall guard2 current register +#define STALL_GUARD_FILTER_ENABLED 0x10000ul +#define STALL_GUARD_TRESHHOLD_VALUE_PATTERN 0x17F00ul +#define CURRENT_SCALING_PATTERN 0x1Ful +#define STALL_GUARD_CONFIG_PATTERN 0x17F00ul +#define STALL_GUARD_VALUE_PATTERN 0x7F00ul + +//definitions for the input from the TCM260 +#define STATUS_STALL_GUARD_STATUS 0x1ul +#define STATUS_OVER_TEMPERATURE_SHUTDOWN 0x2ul +#define STATUS_OVER_TEMPERATURE_WARNING 0x4ul +#define STATUS_SHORT_TO_GROUND_A 0x8ul +#define STATUS_SHORT_TO_GROUND_B 0x10ul +#define STATUS_OPEN_LOAD_A 0x20ul +#define STATUS_OPEN_LOAD_B 0x40ul +#define STATUS_STAND_STILL 0x80ul +#define READOUT_VALUE_PATTERN 0xFFC00ul + +//default values +#define INITIAL_MICROSTEPPING 0x3ul //32th microstepping + +//debuging output +//#define DEBUG + +/* + * Constructor + * number_of_steps - the steps per rotation + * cs_pin - the SPI client select pin + * dir_pin - the pin where the direction pin is connected + * step_pin - the pin where the step pin is connected + */ +TMC26XStepper::TMC26XStepper(int number_of_steps, int cs_pin, int dir_pin, int step_pin, unsigned int current, unsigned int resistor) +{ + //we are not started yet + started=false; + //by default cool step is not enabled + cool_step_enabled=false; + + //save the pins for later use + this->cs_pin=cs_pin; + this->dir_pin=dir_pin; + this->step_pin = step_pin; + + //store the current sense resistor value for later use + this->resistor = resistor; + + //initizalize our status values + this->steps_left = 0; + this->direction = 0; + + //initialize register values + driver_control_register_value=DRIVER_CONTROL_REGISTER | INITIAL_MICROSTEPPING; + chopper_config_register=CHOPPER_CONFIG_REGISTER; + + //setting the default register values + driver_control_register_value=DRIVER_CONTROL_REGISTER|INITIAL_MICROSTEPPING; + microsteps = (1 << INITIAL_MICROSTEPPING); + chopper_config_register=CHOPPER_CONFIG_REGISTER; + cool_step_register_value=COOL_STEP_REGISTER; + stall_guard2_current_register_value=STALL_GUARD2_LOAD_MEASURE_REGISTER; + driver_configuration_register_value = DRIVER_CONFIG_REGISTER | READ_STALL_GUARD_READING; + + //set the current + setCurrent(current); + //set to a conservative start value + setConstantOffTimeChopper(7, 54, 13,12,1); + //set a nice microstepping value + setMicrosteps(DEFAULT_MICROSTEPPING_VALUE); + //save the number of steps + this->number_of_steps = number_of_steps; +} + + +/* + * start & configure the stepper driver + * just must be called. + */ +void TMC26XStepper::start() { + +#ifdef DEBUG + Serial.println("TMC26X stepper library"); + Serial.print("CS pin: "); + Serial.println(cs_pin); + Serial.print("DIR pin: "); + Serial.println(dir_pin); + Serial.print("STEP pin: "); + Serial.println(step_pin); + Serial.print("current scaling: "); + Serial.println(current_scaling,DEC); +#endif + //set the pins as output & its initial value + pinMode(step_pin, OUTPUT); + pinMode(dir_pin, OUTPUT); + pinMode(cs_pin, OUTPUT); + digitalWrite(step_pin, LOW); + digitalWrite(dir_pin, LOW); + digitalWrite(cs_pin, HIGH); + + //configure the SPI interface + SPI.setBitOrder(MSBFIRST); + SPI.setClockDivider(SPI_CLOCK_DIV8); + //todo this does not work reliably - find a way to foolprof set it (e.g. while communicating + //SPI.setDataMode(SPI_MODE3); + SPI.begin(); + + //set the initial values + send262(driver_control_register_value); + send262(chopper_config_register); + send262(cool_step_register_value); + send262(stall_guard2_current_register_value); + send262(driver_configuration_register_value); + + //save that we are in running mode + started=true; +} + +/* + Mark the driver as unstarted to be able to start it again + */ +void TMC26XStepper::un_start() { + started=false; +} + + +/* + Sets the speed in revs per minute + +*/ +void TMC26XStepper::setSpeed(unsigned int whatSpeed) +{ + this->speed = whatSpeed; + this->step_delay = (60UL * 1000UL * 1000UL) / ((unsigned long)this->number_of_steps * (unsigned long)whatSpeed * (unsigned long)this->microsteps); +#ifdef DEBUG + Serial.print("Step delay in micros: "); + Serial.println(this->step_delay); +#endif + //update the next step time + this->next_step_time = this->last_step_time+this->step_delay; + +} + +unsigned int TMC26XStepper::getSpeed(void) { + return this->speed; +} + +/* + Moves the motor steps_to_move steps. If the number is negative, + the motor moves in the reverse direction. + */ +char TMC26XStepper::step(int steps_to_move) +{ + if (this->steps_left==0) { + this->steps_left = abs(steps_to_move); // how many steps to take + + // determine direction based on whether steps_to_mode is + or -: + if (steps_to_move > 0) { + this->direction = 1; + } else if (steps_to_move < 0) { + this->direction = 0; + } + return 0; + } else { + return -1; + } +} + +char TMC26XStepper::move(void) { + // decrement the number of steps, moving one step each time: + if(this->steps_left>0) { + unsigned long time = micros(); + // move only if the appropriate delay has passed: + if (time >= this->next_step_time) { + // increment or decrement the step number, + // depending on direction: + if (this->direction == 1) { + digitalWrite(step_pin, HIGH); + } else { + digitalWrite(dir_pin, HIGH); + digitalWrite(step_pin, HIGH); + } + // get the timeStamp of when you stepped: + this->last_step_time = time; + this->next_step_time = time+this->step_delay; + // decrement the steps left: + steps_left--; + //disable the step & dir pins + digitalWrite(step_pin, LOW); + digitalWrite(dir_pin, LOW); + } + return -1; + } + return 0; +} + +char TMC26XStepper::isMoving(void) { + return (this->steps_left>0); +} + +unsigned int TMC26XStepper::getStepsLeft(void) { + return this->steps_left; +} + +char TMC26XStepper::stop(void) { + //note to self if the motor is currently moving + char state = isMoving(); + //stop the motor + this->steps_left = 0; + this->direction = 0; + //return if it was moving + return state; +} + +void TMC26XStepper::setCurrent(unsigned int current) { + unsigned char current_scaling = 0; + //calculate the current scaling from the max current setting (in mA) + double mASetting = (double)current; + double resistor_value = (double) this->resistor; + // remove vesense flag + this->driver_configuration_register_value &= ~(VSENSE); + //this is derrived from I=(cs+1)/32*(Vsense/Rsense) + //leading to cs = CS = 32*R*I/V (with V = 0,31V oder 0,165V and I = 1000*current) + //with Rsense=0,15 + //for vsense = 0,310V (VSENSE not set) + //or vsense = 0,165V (VSENSE set) + current_scaling = (byte)((resistor_value*mASetting*32.0/(0.31*1000.0*1000.0))-0.5); //theoretically - 1.0 for better rounding it is 0.5 + + //check if the current scalingis too low + if (current_scaling<16) { + //set the csense bit to get a use half the sense voltage (to support lower motor currents) + this->driver_configuration_register_value |= VSENSE; + //and recalculate the current setting + current_scaling = (byte)((resistor_value*mASetting*32.0/(0.165*1000.0*1000.0))-0.5); //theoretically - 1.0 for better rounding it is 0.5 +#ifdef DEBUG + Serial.print("CS (Vsense=1): "); + Serial.println(current_scaling); + } else { + Serial.print("CS: "); + Serial.println(current_scaling); +#endif + } + + //do some sanity checks + if (current_scaling>31) { + current_scaling=31; + } + //delete the old value + stall_guard2_current_register_value &= ~(CURRENT_SCALING_PATTERN); + //set the new current scaling + stall_guard2_current_register_value |= current_scaling; + //if started we directly send it to the motor + if (started) { + send262(driver_configuration_register_value); + send262(stall_guard2_current_register_value); + } +} + +unsigned int TMC26XStepper::getCurrent(void) { + //we calculate the current according to the datasheet to be on the safe side + //this is not the fastest but the most accurate and illustrative way + double result = (double)(stall_guard2_current_register_value & CURRENT_SCALING_PATTERN); + double resistor_value = (double)this->resistor; + double voltage = (driver_configuration_register_value & VSENSE)? 0.165:0.31; + result = (result+1.0)/32.0*voltage/resistor_value*1000.0*1000.0; + return (unsigned int)result; +} + +void TMC26XStepper::setStallGuardThreshold(char stall_guard_threshold, char stall_guard_filter_enabled) { + if (stall_guard_threshold<-64) { + stall_guard_threshold = -64; + //We just have 5 bits + } else if (stall_guard_threshold > 63) { + stall_guard_threshold = 63; + } + //add trim down to 7 bits + stall_guard_threshold &=0x7f; + //delete old stall guard settings + stall_guard2_current_register_value &= ~(STALL_GUARD_CONFIG_PATTERN); + if (stall_guard_filter_enabled) { + stall_guard2_current_register_value |= STALL_GUARD_FILTER_ENABLED; + } + //Set the new stall guard threshold + stall_guard2_current_register_value |= (((unsigned long)stall_guard_threshold << 8) & STALL_GUARD_CONFIG_PATTERN); + //if started we directly send it to the motor + if (started) { + send262(stall_guard2_current_register_value); + } +} + +char TMC26XStepper::getStallGuardThreshold(void) { + unsigned long stall_guard_threshold = stall_guard2_current_register_value & STALL_GUARD_VALUE_PATTERN; + //shift it down to bit 0 + stall_guard_threshold >>=8; + //convert the value to an int to correctly handle the negative numbers + char result = stall_guard_threshold; + //check if it is negative and fill it up with leading 1 for proper negative number representation + if (result & _BV(6)) { + result |= 0xC0; + } + return result; +} + +char TMC26XStepper::getStallGuardFilter(void) { + if (stall_guard2_current_register_value & STALL_GUARD_FILTER_ENABLED) { + return -1; + } else { + return 0; + } +} +/* + * Set the number of microsteps per step. + * 0,2,4,8,16,32,64,128,256 is supported + * any value in between will be mapped to the next smaller value + * 0 and 1 set the motor in full step mode + */ +void TMC26XStepper::setMicrosteps(int number_of_steps) { + long setting_pattern; + //poor mans log + if (number_of_steps>=256) { + setting_pattern=0; + microsteps=256; + } else if (number_of_steps>=128) { + setting_pattern=1; + microsteps=128; + } else if (number_of_steps>=64) { + setting_pattern=2; + microsteps=64; + } else if (number_of_steps>=32) { + setting_pattern=3; + microsteps=32; + } else if (number_of_steps>=16) { + setting_pattern=4; + microsteps=16; + } else if (number_of_steps>=8) { + setting_pattern=5; + microsteps=8; + } else if (number_of_steps>=4) { + setting_pattern=6; + microsteps=4; + } else if (number_of_steps>=2) { + setting_pattern=7; + microsteps=2; + //1 and 0 lead to full step + } else if (number_of_steps<=1) { + setting_pattern=8; + microsteps=1; + } +#ifdef DEBUG + Serial.print("Microstepping: "); + Serial.println(microsteps); +#endif + //delete the old value + this->driver_control_register_value &=0xFFFF0ul; + //set the new value + this->driver_control_register_value |=setting_pattern; + + //if started we directly send it to the motor + if (started) { + send262(driver_control_register_value); + } + //recalculate the stepping delay by simply setting the speed again + this->setSpeed(this->speed); +} + +/* + * returns the effective number of microsteps at the moment + */ +int TMC26XStepper::getMicrosteps(void) { + return microsteps; +} + +/* + * constant_off_time: The off time setting controls the minimum chopper frequency. + * For most applications an off time within the range of 5μs to 20μs will fit. + * 2...15: off time setting + * + * blank_time: Selects the comparator blank time. This time needs to safely cover the switching event and the + * duration of the ringing on the sense resistor. For + * 0: min. setting 3: max. setting + * + * fast_decay_time_setting: Fast decay time setting. With CHM=1, these bits control the portion of fast decay for each chopper cycle. + * 0: slow decay only + * 1...15: duration of fast decay phase + * + * sine_wave_offset: Sine wave offset. With CHM=1, these bits control the sine wave offset. + * A positive offset corrects for zero crossing error. + * -3..-1: negative offset 0: no offset 1...12: positive offset + * + * use_current_comparator: Selects usage of the current comparator for termination of the fast decay cycle. + * If current comparator is enabled, it terminates the fast decay cycle in case the current + * reaches a higher negative value than the actual positive value. + * 1: enable comparator termination of fast decay cycle + * 0: end by time only + */ +void TMC26XStepper::setConstantOffTimeChopper(char constant_off_time, char blank_time, char fast_decay_time_setting, char sine_wave_offset, unsigned char use_current_comparator) { + //perform some sanity checks + if (constant_off_time<2) { + constant_off_time=2; + } else if (constant_off_time>15) { + constant_off_time=15; + } + //save the constant off time + this->constant_off_time = constant_off_time; + char blank_value; + //calculate the value acc to the clock cycles + if (blank_time>=54) { + blank_value=3; + } else if (blank_time>=36) { + blank_value=2; + } else if (blank_time>=24) { + blank_value=1; + } else { + blank_value=0; + } + if (fast_decay_time_setting<0) { + fast_decay_time_setting=0; + } else if (fast_decay_time_setting>15) { + fast_decay_time_setting=15; + } + if (sine_wave_offset < -3) { + sine_wave_offset = -3; + } else if (sine_wave_offset>12) { + sine_wave_offset = 12; + } + //shift the sine_wave_offset + sine_wave_offset +=3; + + //calculate the register setting + //first of all delete all the values for this + chopper_config_register &= ~((1<<12) | BLANK_TIMING_PATTERN | HYSTERESIS_DECREMENT_PATTERN | HYSTERESIS_LOW_VALUE_PATTERN | HYSTERESIS_START_VALUE_PATTERN | T_OFF_TIMING_PATERN); + //set the constant off pattern + chopper_config_register |= CHOPPER_MODE_T_OFF_FAST_DECAY; + //set the blank timing value + chopper_config_register |= ((unsigned long)blank_value) << BLANK_TIMING_SHIFT; + //setting the constant off time + chopper_config_register |= constant_off_time; + //set the fast decay time + //set msb + chopper_config_register |= (((unsigned long)(fast_decay_time_setting & 0x8))<<HYSTERESIS_DECREMENT_SHIFT); + //other bits + chopper_config_register |= (((unsigned long)(fast_decay_time_setting & 0x7))<<HYSTERESIS_START_VALUE_SHIFT); + //set the sine wave offset + chopper_config_register |= (unsigned long)sine_wave_offset << HYSTERESIS_LOW_SHIFT; + //using the current comparator? + if (!use_current_comparator) { + chopper_config_register |= (1<<12); + } + //if started we directly send it to the motor + if (started) { + send262(driver_control_register_value); + } +} + +/* + * constant_off_time: The off time setting controls the minimum chopper frequency. + * For most applications an off time within the range of 5μs to 20μs will fit. + * 2...15: off time setting + * + * blank_time: Selects the comparator blank time. This time needs to safely cover the switching event and the + * duration of the ringing on the sense resistor. For + * 0: min. setting 3: max. setting + * + * hysteresis_start: Hysteresis start setting. Please remark, that this value is an offset to the hysteresis end value HEND. + * 1...8 + * + * hysteresis_end: Hysteresis end setting. Sets the hysteresis end value after a number of decrements. Decrement interval time is controlled by HDEC. + * The sum HSTRT+HEND must be <16. At a current setting CS of max. 30 (amplitude reduced to 240), the sum is not limited. + * -3..-1: negative HEND 0: zero HEND 1...12: positive HEND + * + * hysteresis_decrement: Hysteresis decrement setting. This setting determines the slope of the hysteresis during on time and during fast decay time. + * 0: fast decrement 3: very slow decrement + */ + +void TMC26XStepper::setSpreadCycleChopper(char constant_off_time, char blank_time, char hysteresis_start, char hysteresis_end, char hysteresis_decrement) { + //perform some sanity checks + if (constant_off_time<2) { + constant_off_time=2; + } else if (constant_off_time>15) { + constant_off_time=15; + } + //save the constant off time + this->constant_off_time = constant_off_time; + char blank_value; + //calculate the value acc to the clock cycles + if (blank_time>=54) { + blank_value=3; + } else if (blank_time>=36) { + blank_value=2; + } else if (blank_time>=24) { + blank_value=1; + } else { + blank_value=0; + } + if (hysteresis_start<1) { + hysteresis_start=1; + } else if (hysteresis_start>8) { + hysteresis_start=8; + } + hysteresis_start--; + + if (hysteresis_end < -3) { + hysteresis_end = -3; + } else if (hysteresis_end>12) { + hysteresis_end = 12; + } + //shift the hysteresis_end + hysteresis_end +=3; + + if (hysteresis_decrement<0) { + hysteresis_decrement=0; + } else if (hysteresis_decrement>3) { + hysteresis_decrement=3; + } + + //first of all delete all the values for this + chopper_config_register &= ~(CHOPPER_MODE_T_OFF_FAST_DECAY | BLANK_TIMING_PATTERN | HYSTERESIS_DECREMENT_PATTERN | HYSTERESIS_LOW_VALUE_PATTERN | HYSTERESIS_START_VALUE_PATTERN | T_OFF_TIMING_PATERN); + + //set the blank timing value + chopper_config_register |= ((unsigned long)blank_value) << BLANK_TIMING_SHIFT; + //setting the constant off time + chopper_config_register |= constant_off_time; + //set the hysteresis_start + chopper_config_register |= ((unsigned long)hysteresis_start) << HYSTERESIS_START_VALUE_SHIFT; + //set the hysteresis end + chopper_config_register |= ((unsigned long)hysteresis_end) << HYSTERESIS_LOW_SHIFT; + //set the hystereis decrement + chopper_config_register |= ((unsigned long)blank_value) << BLANK_TIMING_SHIFT; + //if started we directly send it to the motor + if (started) { + send262(driver_control_register_value); + } +} + +/* + * In a constant off time chopper scheme both coil choppers run freely, i.e. are not synchronized. + * The frequency of each chopper mainly depends on the coil current and the position dependant motor coil inductivity, thus it depends on the microstep position. + * With some motors a slightly audible beat can occur between the chopper frequencies, especially when they are near to each other. This typically occurs at a + * few microstep positions within each quarter wave. This effect normally is not audible when compared to mechanical noise generated by ball bearings, etc. + * Further factors which can cause a similar effect are a poor layout of sense resistor GND connection. + * Hint: A common factor, which can cause motor noise, is a bad PCB layout causing coupling of both sense resistor voltages + * (please refer to sense resistor layout hint in chapter 8.1). + * In order to minimize the effect of a beat between both chopper frequencies, an internal random generator is provided. + * It modulates the slow decay time setting when switched on by the RNDTF bit. The RNDTF feature further spreads the chopper spectrum, + * reducing electromagnetic emission on single frequencies. + */ +void TMC26XStepper::setRandomOffTime(char value) { + if (value) { + chopper_config_register |= RANDOM_TOFF_TIME; + } else { + chopper_config_register &= ~(RANDOM_TOFF_TIME); + } + //if started we directly send it to the motor + if (started) { + send262(driver_control_register_value); + } +} + +void TMC26XStepper::setCoolStepConfiguration(unsigned int lower_SG_threshold, unsigned int SG_hysteresis, unsigned char current_decrement_step_size, + unsigned char current_increment_step_size, unsigned char lower_current_limit) { + //sanitize the input values + if (lower_SG_threshold>480) { + lower_SG_threshold = 480; + } + //divide by 32 + lower_SG_threshold >>=5; + if (SG_hysteresis>480) { + SG_hysteresis=480; + } + //divide by 32 + SG_hysteresis >>=5; + if (current_decrement_step_size>3) { + current_decrement_step_size=3; + } + if (current_increment_step_size>3) { + current_increment_step_size=3; + } + if (lower_current_limit>1) { + lower_current_limit=1; + } + //store the lower level in order to enable/disable the cool step + this->cool_step_lower_threshold=lower_SG_threshold; + //if cool step is not enabled we delete the lower value to keep it disabled + if (!this->cool_step_enabled) { + lower_SG_threshold=0; + } + //the good news is that we can start with a complete new cool step register value + //and simply set the values in the register + cool_step_register_value = ((unsigned long)lower_SG_threshold) | (((unsigned long)SG_hysteresis)<<8) | (((unsigned long)current_decrement_step_size)<<5) + | (((unsigned long)current_increment_step_size)<<13) | (((unsigned long)lower_current_limit)<<15) + //and of course we have to include the signature of the register + | COOL_STEP_REGISTER; + //Serial.println(cool_step_register_value,HEX); + if (started) { + send262(cool_step_register_value); + } +} + +void TMC26XStepper::setCoolStepEnabled(boolean enabled) { + //simply delete the lower limit to disable the cool step + cool_step_register_value &= ~SE_MIN_PATTERN; + //and set it to the proper value if cool step is to be enabled + if (enabled) { + cool_step_register_value |=this->cool_step_lower_threshold; + } + //and save the enabled status + this->cool_step_enabled = enabled; + //save the register value + if (started) { + send262(cool_step_register_value); + } +} + +boolean TMC26XStepper::isCoolStepEnabled(void) { + return this->cool_step_enabled; +} + +unsigned int TMC26XStepper::getCoolStepLowerSgThreshold() { + //we return our internally stored value - in order to provide the correct setting even if cool step is not enabled + return this->cool_step_lower_threshold<<5; +} + +unsigned int TMC26XStepper::getCoolStepUpperSgThreshold() { + return (unsigned char)((cool_step_register_value & SE_MAX_PATTERN)>>8)<<5; +} + +unsigned char TMC26XStepper::getCoolStepCurrentIncrementSize() { + return (unsigned char)((cool_step_register_value & CURRENT_DOWN_STEP_SPEED_PATTERN)>>13); +} + +unsigned char TMC26XStepper::getCoolStepNumberOfSGReadings() { + return (unsigned char)((cool_step_register_value & SE_CURRENT_STEP_WIDTH_PATTERN)>>5); +} + +unsigned char TMC26XStepper::getCoolStepLowerCurrentLimit() { + return (unsigned char)((cool_step_register_value & MINIMUM_CURRENT_FOURTH)>>15); +} + +void TMC26XStepper::setEnabled(boolean enabled) { + //delete the t_off in the chopper config to get sure + chopper_config_register &= ~(T_OFF_PATTERN); + if (enabled) { + //and set the t_off time + chopper_config_register |= this->constant_off_time; + } + //if not enabled we don't have to do anything since we already delete t_off from the register + if (started) { + send262(chopper_config_register); + } +} + +boolean TMC26XStepper::isEnabled() { + if (chopper_config_register & T_OFF_PATTERN) { + return true; + } else { + return false; + } +} + +/* + * reads a value from the TMC26X status register. The value is not obtained directly but can then + * be read by the various status routines. + * + */ +void TMC26XStepper::readStatus(char read_value) { + unsigned long old_driver_configuration_register_value = driver_configuration_register_value; + //reset the readout configuration + driver_configuration_register_value &= ~(READ_SELECTION_PATTERN); + //this now equals TMC26X_READOUT_POSITION - so we just have to check the other two options + if (read_value == TMC26X_READOUT_STALLGUARD) { + driver_configuration_register_value |= READ_STALL_GUARD_READING; + } else if (read_value == TMC26X_READOUT_CURRENT) { + driver_configuration_register_value |= READ_STALL_GUARD_AND_COOL_STEP; + } + //all other cases are ignored to prevent funny values + //check if the readout is configured for the value we are interested in + if (driver_configuration_register_value!=old_driver_configuration_register_value) { + //because then we need to write the value twice - one time for configuring, second time to get the value, see below + send262(driver_configuration_register_value); + } + //write the configuration to get the last status + send262(driver_configuration_register_value); +} + +int TMC26XStepper::getMotorPosition(void) { + //we read it out even if we are not started yet - perhaps it is useful information for somebody + readStatus(TMC26X_READOUT_POSITION); + return getReadoutValue(); +} + +//reads the stall guard setting from last status +//returns -1 if stallguard information is not present +int TMC26XStepper::getCurrentStallGuardReading(void) { + //if we don't yet started there cannot be a stall guard value + if (!started) { + return -1; + } + //not time optimal, but solution optiomal: + //first read out the stall guard value + readStatus(TMC26X_READOUT_STALLGUARD); + return getReadoutValue(); +} + +unsigned char TMC26XStepper::getCurrentCSReading(void) { + //if we don't yet started there cannot be a stall guard value + if (!started) { + return 0; + } + //not time optimal, but solution optiomal: + //first read out the stall guard value + readStatus(TMC26X_READOUT_CURRENT); + return (getReadoutValue() & 0x1f); +} + +unsigned int TMC26XStepper::getCurrentCurrent(void) { + double result = (double)getCurrentCSReading(); + double resistor_value = (double)this->resistor; + double voltage = (driver_configuration_register_value & VSENSE)? 0.165:0.31; + result = (result+1.0)/32.0*voltage/resistor_value*1000.0*1000.0; + return (unsigned int)result; +} + +/* + return true if the stallguard threshold has been reached +*/ +boolean TMC26XStepper::isStallGuardOverThreshold(void) { + if (!this->started) { + return false; + } + return (driver_status_result & STATUS_STALL_GUARD_STATUS); +} + +/* + returns if there is any over temperature condition: + OVER_TEMPERATURE_PREWARING if pre warning level has been reached + OVER_TEMPERATURE_SHUTDOWN if the temperature is so hot that the driver is shut down + Any of those levels are not too good. +*/ +char TMC26XStepper::getOverTemperature(void) { + if (!this->started) { + return 0; + } + if (driver_status_result & STATUS_OVER_TEMPERATURE_SHUTDOWN) { + return TMC26X_OVERTEMPERATURE_SHUTDOWN; + } + if (driver_status_result & STATUS_OVER_TEMPERATURE_WARNING) { + return TMC26X_OVERTEMPERATURE_PREWARING; + } + return 0; +} + +//is motor channel A shorted to ground +boolean TMC26XStepper::isShortToGroundA(void) { + if (!this->started) { + return false; + } + return (driver_status_result & STATUS_SHORT_TO_GROUND_A); +} + +//is motor channel B shorted to ground +boolean TMC26XStepper::isShortToGroundB(void) { + if (!this->started) { + return false; + } + return (driver_status_result & STATUS_SHORT_TO_GROUND_B); +} + +//is motor channel A connected +boolean TMC26XStepper::isOpenLoadA(void) { + if (!this->started) { + return false; + } + return (driver_status_result & STATUS_OPEN_LOAD_A); +} + +//is motor channel B connected +boolean TMC26XStepper::isOpenLoadB(void) { + if (!this->started) { + return false; + } + return (driver_status_result & STATUS_OPEN_LOAD_B); +} + +//is chopper inactive since 2^20 clock cycles - defaults to ~0,08s +boolean TMC26XStepper::isStandStill(void) { + if (!this->started) { + return false; + } + return (driver_status_result & STATUS_STAND_STILL); +} + +//is chopper inactive since 2^20 clock cycles - defaults to ~0,08s +boolean TMC26XStepper::isStallGuardReached(void) { + if (!this->started) { + return false; + } + return (driver_status_result & STATUS_STALL_GUARD_STATUS); +} + +//reads the stall guard setting from last status +//returns -1 if stallguard inforamtion is not present +int TMC26XStepper::getReadoutValue(void) { + return (int)(driver_status_result >> 10); +} + +int TMC26XStepper::getResistor() { + return this->resistor; +} + +boolean TMC26XStepper::isCurrentScalingHalfed() { + if (this->driver_configuration_register_value & VSENSE) { + return true; + } else { + return false; + } +} +/* + version() returns the version of the library: + */ +int TMC26XStepper::version(void) +{ + return 1; +} + +void TMC26XStepper::debugLastStatus() { +#ifdef DEBUG +if (this->started) { + if (this->getOverTemperature()&TMC26X_OVERTEMPERATURE_PREWARING) { + Serial.println("WARNING: Overtemperature Prewarning!"); + } else if (this->getOverTemperature()&TMC26X_OVERTEMPERATURE_SHUTDOWN) { + Serial.println("ERROR: Overtemperature Shutdown!"); + } + if (this->isShortToGroundA()) { + Serial.println("ERROR: SHORT to ground on channel A!"); + } + if (this->isShortToGroundB()) { + Serial.println("ERROR: SHORT to ground on channel A!"); + } + if (this->isOpenLoadA()) { + Serial.println("ERROR: Channel A seems to be unconnected!"); + } + if (this->isOpenLoadB()) { + Serial.println("ERROR: Channel B seems to be unconnected!"); + } + if (this->isStallGuardReached()) { + Serial.println("INFO: Stall Guard level reached!"); + } + if (this->isStandStill()) { + Serial.println("INFO: Motor is standing still."); + } + unsigned long readout_config = driver_configuration_register_value & READ_SELECTION_PATTERN; + int value = getReadoutValue(); + if (readout_config == READ_MICROSTEP_POSTION) { + Serial.print("Microstep postion phase A: "); + Serial.println(value); + } else if (readout_config == READ_STALL_GUARD_READING) { + Serial.print("Stall Guard value:"); + Serial.println(value); + } else if (readout_config == READ_STALL_GUARD_AND_COOL_STEP) { + int stallGuard = value & 0xf; + int current = value & 0x1F0; + Serial.print("Approx Stall Guard: "); + Serial.println(stallGuard); + Serial.print("Current level"); + Serial.println(current); + } + } +#endif +} + +/* + * send register settings to the stepper driver via SPI + * returns the current status + */ +inline void TMC26XStepper::send262(unsigned long datagram) { + unsigned long i_datagram; + + //preserver the previous spi mode + unsigned char oldMode = SPCR & SPI_MODE_MASK; + + //if the mode is not correct set it to mode 3 + if (oldMode != SPI_MODE3) { + SPI.setDataMode(SPI_MODE3); + } + + //select the TMC driver + digitalWrite(cs_pin,LOW); + + //ensure that only valid bist are set (0-19) + //datagram &=REGISTER_BIT_PATTERN; + +#ifdef DEBUG + Serial.print("Sending "); + Serial.println(datagram,HEX); +#endif + + //write/read the values + i_datagram = SPI.transfer((datagram >> 16) & 0xff); + i_datagram <<= 8; + i_datagram |= SPI.transfer((datagram >> 8) & 0xff); + i_datagram <<= 8; + i_datagram |= SPI.transfer((datagram) & 0xff); + i_datagram >>= 4; + +#ifdef DEBUG + Serial.print("Received "); + Serial.println(i_datagram,HEX); + debugLastStatus(); +#endif + //deselect the TMC chip + digitalWrite(cs_pin,HIGH); + + //restore the previous SPI mode if neccessary + //if the mode is not correct set it to mode 3 + if (oldMode != SPI_MODE3) { + SPI.setDataMode(oldMode); + } + + + //store the datagram as status result + driver_status_result = i_datagram; +}
\ No newline at end of file |