Thumbnail

Stepper Servo

by Synthian Inc.

Control up to 127 stepper motors as servos with speed and acceleration.

Requires ARC v21 (Updated 3/24/2025)

How to add the Stepper Servo robot skill

  1. Load the most recent release of ARC (Get ARC).
  2. Press the Project tab from the top menu bar in ARC.
  3. Press Add Robot Skill from the button ribbon bar in ARC.
  4. Choose the Servo category tab.
  5. Press the Stepper Servo icon to add the robot skill to your project.

Don't have a robot yet?

Follow the Getting Started Guide to build a robot and use the Stepper Servo robot skill.


How to use the Stepper Servo robot skill

Control up to 127 stepper motors as servos with any stepper motor driver or motor type. This robot skill requires a low-cost and small-profile Arduino connected to each stepper motor driver. Each Arduino is given a unique ID on the optional serial network, which matches an ARC Virtual servo (Vx) port. All 127 Arduinos will share the same PC COM port via the first Arduino's USB connection. An optional end-stop limit switch can calibrate the stepper's home position. For example, if the Arduino firmware bus_id is configured for ID 1, that will match ARC's virtual servo V1.

If you do not want to chain the Arduinos together, you can add multiple instances of this robot skill for each Arduino connected to the PC via USB.

*Note: download the Arduino firmware in the Arduino Firmware section in this manual

Stepper motors consume high resources by depending on several digital I/O working together at the same time in a specific pattern. The pattern and timing is necessary for the stepper to function and not be damaged. Because of this, having a dedicated ardunio or microcontrollers is required.

Operating Modes

Each Arduino's firmware has a configuration for specifying the operating mode. There are four modes: SLAVE_SERIAL0, SLAVE_SERIAL1, MASTER_SERIAL0, and MASTER_FORWARDING. They are explained below, and only one of the modes can be active based on your configuration and hardware.

MASTER_SERIAL0 This mode uses a single Arduino over USB with no additional Arduinos connected to the serial network. There is no serial network for this setting. This is for a single Arduino with one stepper.

Enable MASTER_SERIAL0 if all of these conditions are met...

  • this is the only Arduino that will be connected to the PC (i.e., you only need one stepper motor)
  • this Arduino is connected to the PC via the USB cable

    User-inserted image

MASTER_FORWARDING This mode configures this Arduino as a forwarder for multiple Arduinos on the serial network. This master Arduino will be connected to the PC via the USB cable. All slave Arduinos will link to this Arduino's Serial1 port (TX1/RX1) with diodes for each Arduino.

*Disclaimer: the serial network may not work with all Arduino. You may also not mix 5v and 3.3v Arduino on the serial network.

Enable MASTER_FORWARDING if all of these conditions are met...

  • this Arduino is the master with BUS_ID 1
  • this Arduino has more than 1 Serial port (i.e., it has RX1/TX1 pins)
  • there are other Arduinos connected to the RX1/TX1 of this master Arduino in a serial network

*Note: If you use FORWARDING mode and receive an error about Serial1 not being defined, then this Arduino does not qualify for this mode.

User-inserted image

SLAVE_SERIAL0 This Arduino's RX0/TX0 pins are connected to a master Arduino's RX1/TX1 pins as part of the serial network.

Use SLAVE_SERIAL0 mode if all of these conditions are met...

  • this Arduino is part of the serial network as a slave (it is not a master)
  • this Arduino has onboard RX0/TX0 pins, and they're connected to the master Arduino's RX1/TX1 pins
  • there is a diode on this Arduino's TX0 pin that is connected to the master Arduino's RX1 pin

    User-inserted image

SLAVE_SERIAL1 This Arduino's RX1/TX1 pins are connected to a master Arduino's RX1/TX1 pins as part of the serial network.

Use SLAVE_SERIAL1 mode if all of these conditions are met...

  • this Arduino is part of the serial network as a slave (it is not a master)
  • this Arduino has onboard RX1/TX1 pins, and they're connected to the master Arduino's RX1/TX1 pins
  • there is a diode on this Arduino's TX1 pin that is connected to the master Arduino's RX1 pin

*Note: If you use SLAVE_SERIAL1 mode and receive an error about Serial1 not being defined, it means this Arduino does not meet the requirements for this mode.

User-inserted image

Serial Network (optional)

Optionally, you can connect multiple Arduinos in a serial network to share the same USB connection. This requires the first Arduino (the master) to have two serial ports. You can recognize if the first Arduino has two serial ports by the presence of the RX1 and TX1 pins.

*Note: The firmware has documentation explaining various modes for which the Arduino can be configured.

This diagram demonstrates how multiple Arduinos can be connected to a Master Arduino with a diode. Ensure the diode's band is facing the correct direction. The 100k ohm resistor holds the RX1 high because the diodes prevent the slave TX lines from doing so.

User-inserted image


Configuration

User-inserted image

The configuration window has many settings. The settings are described below.

1) Port Select the COM port for the Arduino 1 wire serial network interface.

2) DTR Enabled Some Arduinos (e.g., Leonardo, Pro Micro) require this setting to be enabled for communication. If your Arduino is unresponsive when querying Ping or getting a servo position, this may need to be enabled. It is enabled by default.

3) Virtual Port Selections Put a checkbox next to each Virtual Port you wish to bind to the stepper motors. For example, BUS_ID 1 is V1. And BUS_ID 5 is V5, etc. Any commands to move the Vx port will be sent to the respective stepper motor with the matching BUS_ID. Sending a Servo.setPosition(v2, 100) will move the stepper motor with BUS_ID 2 to position 100.

4) Max Steps Each virtual port has a Max Steps setting that determines how many steps a stepper motor will move when given a servo position from ARC.

The ARC Max servo Position setting (found in ARC Project Settings) defines the upper limit of servo positions used to control the stepper. By default, this value is 180, but it can be increased up to 2 billion.

The Max Steps setting defines how many total steps the stepper will move when a position at the ARC Max servo Position value is commanded. For example:

  • If Max servo Position = 180 (default)
  • And Max Steps = 1500

Then:

  • Sending servo position 180 Stepper moves 1500 steps
  • Sending servo position 90 Stepper moves 750 steps
  • Sending servo position 1 Stepper moves 0 steps (minimum position)

Stepper Movement Formula

stepperPosition = Map(servoPosition, arcMinServoPositions, arcMaxServoPositions, 0, maxStepperPositions);

Where:

  • servoPosition is the position value sent to the stepper (1 to Max servo Position)
  • arcMinServoPositions is the minimum value (always 1 in ARC)
  • arcMaxServoPositions is the configured maximum servo value in ARC (default 180)
  • maxStepperPositions is the number of steps this stepper should take at the max position (i.e., the Max Steps setting)

The servo position you send is scaled between 0 and Max Steps, based on the configured Min (1) and Max servo Positions in ARC. This makes it possible to have consistent, precise control over stepper motors using ARC's standard servo interface.


Advanced servo Features

This supports the advanced options of servos for Speed and Acceleration settings. By configuring the acceleration option in ARC's advanced servo settings, the stepper motors will speed up and slow down between their source and destination positions.

When editing a robot skill for the servo configuration, the Advanced button allows you to specify the speed and acceleration. You may also use the respective JavaScript, Python, and EZ-Script commands to determine the speed and acceleration programmatically.

User-inserted image

A lower acceleration value will cause the stepper to accelerate slower as it begins to move. -1 means not changing by maintaining the current setting, and 0 means no acceleration.

User-inserted image

Home Position

The HOME position is servo position 1 in ARC. By default, the position on power-on of all steppers and their respective Arduino's will be servo position 1. The ControlCommand can specify any position to become HOME (position 1). This robot skill has a ControlCommand for setting the HOME position to calibrate the stepper motor positions. Additionally, you can use an end-stop limit switch to calibrate the stepper into the home (position 1). A stepper motor knows its position by how many "steps" have been sent to move it in either direction. During the operation, the Stepper servo Arduino Firmware keeps track of the steps and, therefore, the relative position. However, because the position is arbitrary on power-on, you may need to calibrate it using an Endstop Limit Switch.

Endstop Limit Switch

This robot skill supports an optional limit switch connected to the Arduino for calibrating the position of the stepper motor to become servo Position 1. When the switch is closed to GND, the stepper will automatically be configured to be in servo position 1 in ARC. When the end limit switch is shorted, this is similar to sending the SetAsHome ControlCommand to the Arduino for the specified virtual port. The Arduino firmware can be edited to include the option of auto-calibrating when power is applied. Otherwise, you can programmatically enable calibration with the control command.

Place a limit switch on the lowest position of the lever connected to the stepper motor. If configured, this switch will be used as the limit switch. When calibrating, the stepper will move toward the lowest position until the limit switch is reached. Once the limit switch is reached, that will be position 1 in ARC until the power is cycled. There is no memory of the position between power cycles. The lowest position means the stepper begins rotating by decreasing from the current position.

Ensure the calibration rotates your stepper in the correct direction, which will be position 1 in ARC. If the stepper rotates in the wrong direction, you may need to specify the INVERT option in the firmware. The limit switch should always be at the lowest position for the stepper, which means position 1 in ARC.

Increase servo Resolution

ARC has a project-wide setting for increasing the servo resolution across all robot skills. Increasing this value will give the stepper a high resolution supported by your stepper motor.

User-inserted image

Test Bi-direction Communication

This allows the Arduino to report the current position of the stepper during movement using the script command GetPositionRealtime(). This command will retrieve the real-time position of the stepper from the respective Arduino. The Arduino's are responsible for moving the stepper, including acceleration and speed.

The robot skill main screen has a Ping menu option to test bi-directional communication.

User-inserted image

When the ping option is selected, a new window will open with options to ping the Arduinos and view their firmware version. Here, you can choose the port for each Arduino on the serial network to verify communication and the version number.

User-inserted image

Control Commands

There are several ControlCommand() for interacting with the stepper servo robot skill.

ControlCommand("Stepper Servo", "Calibrate", "ServoPort"); This ControlCommand takes the virtual servo port as the parameter. It will instruct the Arduino with the ARC Stepper servo firmware to move the stepper until the End Limit Switch is triggered. This command requires an end-limit switch; otherwise, the stepper will rotate indefinitely. Refer to the Limit Switch section in this manual to read more about a limit switch and how to configure it. The servo port is the virtual servo that matches the ID; for example, if the Arduino firmware is configured for ID1, this will be V1.

ControlCommand("Stepper Servo", "SetAsHome", "ServoPort"); The SetAsHome command will set the current position of the stepper as servo position 1. For example, if you move the servo into a position (i.e., 90) and then specify the SetAsHome control command, the current position will become the new servo position 1 for that virtual port. servo position 1 is as low as a servo can move, so when you send SetAsHome to the specified virtual port, it will now refer to the current position as 1. The servo port is the virtual servo that matches the ID; for example, if the Arduino firmware is configured for ID1, this will be V1.

Arduino Firmware

The Arduino code must be downloaded from here and programmed onto each Arduino. The recommended Arduino for this usage would be small and affordable, such as a Pro Micro, Micro, Nano, or ATTiny. These are all very low-cost devices that are small and do not take up much space.

  1. Download Arduino Firmware from here: Stepper_Servo (Version 20230223).zip (Updated Feb 23, 2023)

  2. Uncompress the firmware to a folder and open the INO file with the Arduino editor

    User-inserted image

  3. Configure the BUS_ID on the one-wire serial network in the Arduino editor for this instance. It can be a number between 1 and 127. This number will correspond with the respective Virtual servo Port in ARC. Each Arduino/Stepper will have a unique ID. Remember, the ID corresponds to ARC virtual servo numbers. So BUS_ID 2 is ARC's servo port V2.

    User-inserted image

  4. In the Arduino editor, configure the type of stepper driver that you will have connected to the Arduino. There are multiple types of stepper drivers to be supported, including two-wire, three-wire, four-wire, etc. Uncomment the section of the kind you will use with this Arduino. Only one area can be uncommented at a time, so comment out another section if editing a previous configuration.

    User-inserted image

  5. In the Arduino editor, you can configure control line wires from the Arduino to the Stepper Driver if they need to be inverted. This is only if the stepper driver requires an inverted GPIO so that HIGH becomes LOW and vice versa. 

    User-inserted image

  6. Ensure you have selected the correct Arduino Board that you are using

    User-inserted image

  7. Program the Arduino with the edited code.

    User-inserted image

Troubleshooting

  • Ensure you have the latest Arduino firmware installed on your Arduino(s)

  • Ensure you have the latest Stepper servo robot skill installed in ARC

  • Ensure you have the latest ARC installed (Early Access edition is recommended)

  • The stepper driver needs to be connected to the Arduino I/O pins. Ensure the pins are connected and specified in the Arduino firmware. Ensure you uncommented the correct stepper driver type in the Arduino firmware before uploading it to the Arduino.

  • A common GND is necessary across all electronic devices. Ensure the stepper motor driver, stepper motor, Arduino, and PC have a common ground. The Arduino will be grounded to the PC if connected via a USB cable.

  • The configured BUS_ID on the Arduino must match the selected checkboxes in the Robot Skill's configuration screen

Remember, the BUS_ID in the firmware will match ARC's Virtual servo (Vx) ports. So BUS_ID 1 will match V1 in ARC. BUS_ID 2 will match V2 in ARC, and so on...


ARC Pro

Upgrade to ARC Pro

ARC Pro will give you immediate updates and new features needed to unleash your robot's potential!

PRO
USA
#105   — Edited

got it working again - forward, and reverse - works well

I tested a Bipolar Stepper, NEMA 17 with L298N H-Bridge with an Arduino with DJ's code

User-inserted image

in code using 8, 9, 10, 11 for pins 1,2,3,4,

external 12 v power supply for stepper motor

all grounded together, the 12v and the Arduino - used a small bread board, EnA EnB to bread board 5v form Arduino

used in the code - #define ACCELERATION 300 - spins better - will try a higher ACCELERATION later

#106   — Edited

Ahh, got it working and it was a stupid mistake. After looking it over a little bit wouldn't it be better to have a built in variable in each stepper panel to get realtime position rather than making a loop in blockly. The Get realtime position could be used as a one shot when a specific position is needed. What is the proper way to make a loop in Blockly to constantly get the realtime position and can that be separate blocks on the sidelines that constantly monitors it?

#107  

Another feature that you may want to put in the stepper panel is rather than using limit switch to make it zero what if you referenced a pot that knows it's position and input that position before it starts up. This value from Adc would be manipulated/multiplied as the starting point. There could be checkbox in the panel if someone wanted to use this or not. If you are using a 180 scale it may start at 122 and be proportional to the total. This is similar to what you are actually doing when you input a value Into the PWM servo feedback.

PRO
USA
#108  

Quote:

you said - Ahh, got it working...
What was your mistake?

#109  

Quote:

wouldn't it be better to have a built-in variable in each stepper panel to get real-time position rather than making a loop in blockly

It was mentioned earlier that it is impossible to have a variable because it is a separate piece of hardware. You can use functions like a variable. A function returns a value. Use the function to get the position instead of using a variable.

Quote:

rather than using limit switch to make it zero what if you referenced a pot that knows it's position and input that position before it starts up.

Combine this robot skill with one of the servo position robot skills that does just that. It was mentioned above as well.

PWM servo feedback (ADC): https://synthiam.com/Support/Skills/Servo/PWM-Servo-Feedback-ADC?id=18047 PWM servo feedback (I2c): https://synthiam.com/Support/Skills/Servo/PWM-Servo-Feedback-i2c?id=18069

PRO
USA
#110   — Edited

I got it working again  as I stated above - forward, and reverse - works well

I tested a Bipolar Stepper, NEMA 17 with L298N H-Bridge with an Arduino with DJ's code

In the code I used 8, 9, 10, 11 for pins 1,2,3,4,

external 12v power supply for stepper motor

all grounded together, the 12v and the Arduino - used a small bread board, EnA EnB to bread board 5v form Arduino

used in the code - #define ACCELERATION 300 - spins better - will try a higher ACCELERATION later

The ping worked, you can see above

Quote:

Quote: you said - Ahh, got it working...  just a      mistake
What was your mistake at first?

EzAng

#111  

I clicked on the green update which I thought would update my Arduino, but that was for ARC. It was when I was looking for troubleshooting that I saw the update for the Arduino. As soon as I saw it I knew what was wrong-downloaded then uploaded, tried it out and it works great.I knew better than that because it was downloaded the first time.

#112   — Edited

Synthiam support, yes you did mention it and I was a bit amped up and forgot because finally got it working again. Will try to combine the skills for the second answer. How would you write the function in Blockly or on main screen to see where it thinks it is and display it. I'm new to writing functions.