Step #1: Pulse Width ModulationPrevNext
Pulse Width Modulation (PWM) is a technique for creating a digital square wave signal. A square wave has 3 main characteristics
Amplitude - The amount the signal changes between On and Off states
Frequency - The number of times the signal repeats in a given time frame
Duty Cycle - The proportion of On time to Off time usually expressed as a percentage
In these digital systems the amplitude is fixed between 0 and 1, between 0V and VCC. In my testing this was +5V.
In it's simplest form a PWM square wave can be created with a few lines of code. The following code would produce a signal with a 1 second Frequency and a 10% duty cycle. This would be functional but the Arduino would be dedicated to producing this signal the whole time and would not do anything else. This is why the blink without delay
example is so important to learn.
With a software generated signal any variation in the code execution time can cause variations in the output. Conditional statements may not execute on every loop so it is hard to guarantee that each loop will take exactly the same amount of time. Modern microcontrollers have dedicated hardware modules to generate a code independent square wave. Values for Duty cycle and Frequency are written to registers within the controller and the hardware generates the appropriate signal on the desired output pin.
The registers for the Hardware PWM module can be written to directly as described here
, but the joy of the Arduino environment is that somebody is likely to have written a library to simplify that process. The TimerOne library
provides control over the HardwarePWM module connected to timer 1 (aptly). This library has easy functions for setting the frequency and duty cycle of the square wave being generated
pwm(pin, duty, period) - Generates a PWM waveform on the specified pin.
The minimum freqeuncy supported by the library is 1ms and the maximum is 8.3s. The duty is specified as a 10 bit value, between 0 and 1023, and this correlates to the 0% and 100% values respectively.
Step #2: PWM for servo controlPrevNext
A servo is a small mechanical actuator that typically consists of a motor a potentiometer and some control electronics. The potentiometer allows the controller to determine what angle the output is at and provide a closed loop servomechanism. The servo controller receives input in the form of a square wave signal so it can drive to the desired angle.
For servo control the Frequency of the signal is fixed, and the duty cycle varies.
Servos accept a common control signal, a square wave with a repeat frequency of 20ms and an on period of 1ms (5%) to 2ms (10%). It is the width of the on pulse that indicates what angle the servo should be at.
Here is the code to set up the two extreme signals using the Timer one libraries
- pwm(9, 51, 20000);
- pwm(9, 102, 20000);
You'll note that using this method there are only 50 steps between the minimum angle and the maximum angle, this is sufficient resolution for most basic systems. There are better methods for controlling the servo to provide more accurate resolutions but they are beyond the scope of this project.
Step #3: PWM for stepper motorsPrevNext
A stepper motor is a brushless DC electric motor that divides a full rotation into a number of steps. This motors position can be determined by counting the number of steps that have been commanded providing open loop control of the system. A stepper motor driver is typically added to the system to power the stepper motor and simplify the control into step and direction pulses. Each pulse on the step line causes the motor to move a step, or part step, in a give direction.
For stepper motor driver control the duty cycle can be fixed and the Frequency varied.
The stepper motor driver expects a series of input pulses to move the motor to any given angle. The driver moves the motor one step for each input pulse. The direction of the motor is set by the second input.
Here is the code to set up the two different speeds using the Timer one libraries
- pwm(9, 512, 20000);
- pwm(9, 512, 40000);
A smaller value for the period produces a higher frequency which gives more pulses per second and makes the motor turn faster. Changing the period while the motor is running can produce acceleration in the motor which will help the stepper reach it's maximum top speed.
Step #4: Putting it all into a projectPrevNext
I discovered all this information with a goal in mind. I wanted to drive the stepper motor connected to the Z axis of my laser cutter. Manually generating a software PWM signal on an Arduino pro mini just wasn't making the axis move fast enough and the occasional jitter in code had a habit of making the motor stall. I needed this clean output signal to drive the motor faster and more reliably, the results were pretty promising.
There is one major drawback to this method of control over stepper motors. A stepper motor uses open loop control to determine where it is, this is done by counting the number of steps the motor has made in any given direction. When using the hardware module to generate the PWM like this, nothing is actually counting the steps. This method of control is only really suitable when there is another feedback mechanism. In my case the Z axis travels to the end switches and stops when it gets there. I don't care how many steps it takes to do that.
I have another project in development that will use this same control method, in that system the user will provide the feedback and will stop the motor when it is in the correct position. This method is not really suitable for controlling the laser cutter as it traces out a design so I will move on to look at the next thing, expect more projects in the near future.