FluidNC supports multiple spindles on one machine. Spindles can be controlled by different hardware interfaces like relays, PWM, DACs, or RS485 serial interfaces to VFDs. Lasers are treated as spindles.
Each spindle is assigned a range of tool numbers. You change spindles by issuing the GCode command "M6 Tn", where n is the tool number. Tool numbers within the assigned range for a given spindle will activate that spindle - and the detailed number within the range could be used to select the specific tool on the spindle. This lets you have, for example, a single machine with an ATC spindle and a laser. A single GCode file could allow you to etch and cut out a part. Most CAM programs support tool numbers. You could also have a gantry with both a low-speed high-torque pulley spindle and also a high-speed direct drive spindle.
0-10V control is designed for spindle controllers that have a 0-10V control input. The signal out of FluidNC is PWM at the pin voltage (typically 3.3v). You must use an adapter to scale it to the voltage you need (typically 10V). The reason for this special case of PWM control is that the VFDs typically have a separate signal input for forward and reverse directions. If you don't need direction control, you should use the basic PWM spindle type.
Example
10V:
forward_pin: gpio.13
reverse_pin: gpio.17
pwm_hz: 5000
output_pin: gpio.4
enable_pin: NO_PIN
direction_pin: NO_PIN
disable_with_s0: false
s0_with_disable: true
spinup_ms: 0
spindown_ms: 0
tool_num: 0
speed_map: 0=0.000% 1000=0.000% 24000=100.000%
off_on_alarm: false
The BESC spindle is designed for use with RC hobby type Brushless Electronic Speed Controllers. These use the same type of PWM signal as an RC servo. These typically only use one I/O pin, but other standard spindle pins can be used. The PWM signal ranges from a min value to a max value, so it is generally outputting at all times.
Special settings
besc:
pwm_hz: 50
output_pin: gpio.4
enable_pin: NO_PIN
direction_pin: NO_PIN
disable_with_s0: false
s0_with_disable: true
spinup_ms: 0
spindown_ms: 0
tool_num: 100
speed_map: 0=0.000% 255=100.000%
min_pulse_us: 900
max_pulse_us: 2200
off_on_alarm: false
This could be used to control a hobby servo in an application like a pen plotter. If you wanted to use an
M3 S255
to move the pen down and anM5
orM3 S0
to lift it.
Thespinup_ms
/spindown_ms
allows to wait for the pen to move before continuing the execution.
Also see the RC servo feature under motors axes.
The DAC uses the ESP32's built in DAC hardware. This can only be used on gpio.25 and gpio.26. It outputs a 0-3.3V analog voltage (not PWM). In most cases a PWM will be better. The DAC resolution is only 8 bit (0-255) and a PWM can be up to 16 bit (0-65535).
example
DAC:
output_pin: gpio.25
enable_pin: NO_PIN
direction_pin: NO_PIN
disable_with_s0: false
s0_with_disable: true
spinup_ms: 0
spindown_ms: 0
tool_num: 100
speed_map: 0=0.000% 255=100.000%
off_on_alarm: false
Unless your controller has built-in RS485 outputs, you must use a UART to RS485 adapter board to turn the serial messages of the controller to RS485 messages and vice versa. UART to RS485 adapter boards are inexpensive, and typically run on either 3.3V or 5V DC.
You must specify the UART parameters in a uart_section of the config file. We strongly recommend you specify this as a separate uart<number>
section and refer to the UART number in the configuration of your spindle using uart_num
. The legacy way of specifying the UART parameters inside the spindle section did not allow you to set the UART number and could cause conflicts if you defined multiple UARTs.
uart1:
txd_pin: gpio.14
rxd_pin: gpio.15
rts_pin: gpio.13
baud: 9600
mode: 8N1
The rts_pin
is used for communications direction control. Some RS485 adapter chips automatically control the direction and do not need an rts_pin
.
You should power your RS485 adapter with 3.3v. Otherwise the return signals to the ESP32 could be damaged.
RS485 terminals can be labeled A and B or + and -. Typically (A connects to -) and (B connects to +), but many people have found that the 2 wires need to be swapped. Often there is a ground wire. Most people have better luck without a ground connection between the controller and VFD. RS485 is a differential signal, so it does not need a ground reference.
You should use a 20 - 22 AWG twisted pair of wires. If the cable is shielded connect the shield wire to ground at one end only.
If you have Tx and Rx LEDs at the controller side the Tx should blink a few times per second. The Rx LED should blink right after. It does so so quickly that they may appear to blink at the same time. If the Rx LED stays on, try swapping A and B at the controller end.
These spindles are controlled with a RS485 connection. Add a top-level definition for one of the supported VFD models to your config file:
Huanyang
YL620
H100
H2A
NowForever
VFDs controlled with RS486 are a little more complicated to set up compared to PWM control, but they offer several advantages. They constantly monitor the spindle. If it appears the spindle is not working at the specified speed, the job will be stopped. It is important that the VFD is fully powered on before the the ESP32 is powered on. Some take several seconds before they are ready to communicate. Do not power cycle the VFD when the ESP32 is on. It may interrupt the communications and cause problems.
We recommend the Huanyang VFD for new users. The FluidNC firmware developers have this VFD and can help debug problems. We cannot help too much with the other types.
Most VFD can also be controlled by a 0-10V signal. This method is easier to debug by new users with a simple voltmeter.
Most problems are due to hardware, wiring, or VFD setup issues. Double check the RS485 wiring. Try swapping the data lines. Sometimes it works better without the ground wire. Be sure the VFD parameters are set correctly and the RS485 is the primary control source. RS485 can be tricky and the FluidNC developers will likely refuse to help with cheap hardware and hacky wiring.
VFD RS485 Unresponsive Message This is normal if your VFD is not connected, not powered on, or connected correctly. RS485 can be complicated for non-engineers. We cannot help you fix this for free (seriously donate big before you ask). Please read the internet on how RS485 works and how to wire it. If you want a simpler interface, try 0-10V control.
Get your VFD and spindle running in manual control mode. It makes no sense to try RS485 until you know the VFD is working correctly. Test everything, like min RPM, max RPM, spin up and down times, forward and reverse.
Wire your RS485 adapter to the VFD. Devices are typically labeled with the symbols A and B or + and -. Wire the same symbols together. If you have A/B at one end and +/- at the other, A goes to - and B goes to +. Wire ground to ground.
If things don't work, try swapping the A/B or +/- wires at one end. If that does not work, try removing the ground connection wire with both combinations of the A/B, +/- wiring.
Double check that you are in RS485 control mode. Check that the baud rate setups match on each end.
This is the standard Huanyang VFD (all power levels). Part numbers start with Hy, like HY02D223B-T. The control is via RS485.
The VFD registers need to be set up prior to use. FluidNC will not change any of the registers. Read your VDF documentation on how to do that. Here are some typical values that work for most spindles.
Register | Value | Description |
---|---|---|
PD001 | 2 | RS485 Control of run command |
PD002 | 2 | RS485 Control of frequency |
PD004 | 400.00 | Base frequency as rated on spindle |
PD005 | 400.00 | Maximum frequency Hz (400Hz * 60sec/min = 24000rpm) |
PD011 | 100.00 | Minimum speed in Hz (Typ. Air cooled 120, Water cooled 100) |
PD014 | 6.0 | Acceleration time in seconds |
PD015 | 6.0 | Deceleration time |
PD023 | 1 | Enable reverse |
PD141 | 220.0 | Spindle Voltage |
PD142 | 3.7 | Max current (typ. 0.8kw=3.7) |
PD143 | 2 | Poles |
PD144 | 3000 | Revolutions at 50Hz |
PD163 | 1 | RS485 Modbus address |
PD164 | 1 | Baud rate of 9600 |
PD165 | 3 | RS485 Mode RTU, 8N1 |
The min and max speeds set in the VFD will be displayed in the startup messages. They might be spread out through the messages, because they are coming from a separate task.
[MSG:INFO: Huanyang PD005,PD011 Freq range (100,400) Hz (6000,24000) RPM]
[MSG:INFO: Huanyang PD144 Rated RPM @ 50Hz:3000]
[MSG:INFO: Huanyang PD143 Poles:2]
[MSG:INFO: Huanyang PD014 Accel:6.000]
[MSG:INFO: Huanyang PD015 Decel:6.000]
A minimum speed is typical with VFD spindles because they lack power and can overheat at lower speeds. A good speed linearization setting for the values above would be...
speed_map: 0=0% 0=25% 6000=25% 24000=100%
The minimum speed of 6000 is 25% of the max speed of 24000. This setting would mean all values between 0-6000 would still result in 6000 RPM.
Check your RS485 adapter documentation for proper wiring methods and connections. Here is some information on the RS485 module.
uart1:
txd_pin: gpio.14
rxd_pin: gpio.15
rts_pin: gpio.13
baud: 9600
mode: 8N1
Huanyang:
uart_num: 1
modbus_id: 1
tool_num: 0
speed_map: 0=0% 0=25% 6000=25% 24000=100%
off_on_alarm: false
The VFD provides feedback to FluidNC. We use this to make sure the spindle is running at the right speed. If it is not running at the speed requested, it will shut down with an alarm after a few seconds. This could be due to many reasons. One reason is that the VFD has a max and min frequency setting. If you request a speed lower than the min it will run at the minimum. The same condition applies to the maximum. Therefore, if you request a speed above 0, but outside the range. The reported speed will not match your requested speed and will shut down. You will get a warning message at the serial console why this is happening. This issue can be fixed by using the correct type of speed_map.
YL620 is a Chinese VFD made by Yalang.
They can be controlled by 0-10V analog or by RS485 (Modbus). The following notes are for RS485 use.
The VFD registers need to be set up prior to use. FluidNC will not change any of the registers. Read your VFD documentation on how to do that. Here are some typical values that work for most spindles.
The Hz values given below indicate the frequency that is sent to the motor. A typical
2-pole motor will rotate once per Hz, so to get RPMs you multiply Hz (cycles/sec) times
60 (sec/min). So a 2-pole motor at 400 Hz runs at 400 * 60 = 24,000 RPM nominally. In
practice it will run slightly slower due a real-world factor called "slip". A 24,000
nominal motor might actually run at 23,500 with no load and 23,000 RPM under load.
Register | Value | Description |
---|---|---|
P00.00 | 4000 | Main frequency in deci-HZ - 4000 is 400.0 Hz |
P00.01 | 3 | Command Source. 3 is for control via RS485 |
P03.00 | 3 | RS485 Baud Rate. 3 is for 9600 |
P03.01 | 1 | Modbus Address. Typically you want to use 1. |
P03.02 | 2 | RS485 Data format. 2 is 8 data bits, 1 stop bit, no parity |
P03.08 | 1000 | Lowest frequency in deci-Hz - 1000 is 100.0Hz |
A minimum speed is typical with VFD spindles because they lack power and can overheat at lower speeds. A good speed linearization setting for the values above would be...
speed_map: 0=0% 0=25% 6000=25% 24000=100%
The minimum speed of 6000 is 25% of the max speed of 24000. This setting would mean all values between 0-6000 would still result in 6000 RPM.
Check your RS485 adapter documentation for proper wiring methods and connections. Here is some information on the RS485 module.
uart1:
txd_pin: gpio.14
rxd_pin: gpio.15
rts_pin: gpio.13
baud: 9600
mode: 8N1
YL620:
uart_num: 1
modbus_id: 1
tool_num: 0
speed_map: 0=0% 0=25% 6000=25% 24000=100%
off_on_alarm: false
Important notes The VFD provides feedback to FluidNC. We use this to make sure the spindle is running at the right speed. If it is not running at the speed requested, it will shut down with an alarm after a few seconds. This could be due to many reasons. One reason is that the VFD has a max and min frequency setting. If you request a speed lower than the min it will run at the minimum. The same condition applies to the maximum. Therefore, if you request a speed above 0, but outside the range. The reported speed will not match your requested speed and will shut down. You will get a warning message at the serial console why this is happening. This issue can be fixed by using the correct type of speed_map.
This is the standard H100 VFD (all power levels). Part numbers typically look like H100-xxx.
The VFD registers need to be set up prior to use. FluidNC will not change any of the registers. Read your VDF documentation on how to do that.
The most relevant sections are:
F011 (min frequency)
F005 (max frequency)
The min and max speeds set in the VFD will be displayed in the startup messages. They might be spread out through the messages, because they are coming from a separate task.
[MSG:INFO: VFD: Max speed:24000rpm]
[MSG:INFO: VFD: Min speed:6000rpm]
A minimum speed is typical with VFD spindles because they lack power and can overheat at lower speeds. For speed_map, feedback and the RS485 adapter settings, see the Huanyang section for more details.
If you don't specify the speed_map, the firmware will automatically put in the default values based on the frequencies that are set in the VFD. Only specify a speed_map if you use a gearbox or other contraption.
uart1:
txd_pin: gpio.26
rxd_pin: gpio.16
rts_pin: gpio.4
baud: 9600
mode: 8N1
H100:
uart_num: 1
modbus_id: 1
tool_num: 0
speed_map: 0=0% 0=25% 6000=25% 24000=100%
The Huanyang P2 series inverters, also dubbed 'H2A/H2B/H2C' or sometimes 'P2A' inverters are supported as well. This product was designed to be the 2nd generation of the popular Huanyang inverter from hy-electrical.com. These VFD's are small, and low power VFD's are usually white or gray. The sticker on the side of the inverter clearly tells that this is the inverter in question.
The manual can unfortunately be a bit confusing at times when it comes to setting up the RS485 connection.
At the top of the box is a RS485 connector with a 4-wire screw terminal. Wiring should use these 4 pin:
Prefer to use a shielded wire for the connector, and never run this wire next to a 220V, stepper or spindle wire. Also, ground one end of the shielding.
ALWAYS read the manual for VFD's! This is imperative to get motor speed etc. all correct. The H2x series inverters use real RPM values, so you need to set these accordingly, or the device won't work correctly. Next to that, you need to set certain settings to get RS485 working correctly, most notably:
Setting | Value | Description |
---|---|---|
F0.02 | 7 | Set RS485 mode |
F0.04 | 2 | Set RS485 mode |
F0.09 | 4 | Set RS485 mode |
F9.00 | 4 | 19200 baud |
F9.01 | 0 | 8,N,1 parity |
F9.02 | 1 | ModBus address |
F9.05 | 0 | Non-standard modbus mode |
F9.07 | 0 | Write operations responded |
We recommend setting 19200,8N1 for this VFD. During spindle sync, there can be quite a bit of communication, and 2400 baud might get you in trouble. 19200 is more than enough for anything you want to throw at a VFD. Using even higher baud rates will probably just result in errors.
Manufacturer: Shenzhen NowForever Electronics Technology CO.,LTD.
Website: http://www.nowforever.cn/
The D series of VFDs can be found in the control box of Chinese CNC6040 cnc routers.
Unfortunately there is no manual for the D series so all of this information is based on the manual for the E series.
However there is a comparison chart made by someone else that shows that almost all parameters are the same.
The manual for the E series can be found online in various places by searching for nowforever e100 manual using your preferred search engine.
Comparison chart (german, also contains link for e series manual): http://moh-computer.de/frequenzumformer-parameter/
The current implementation for supporting NowForever VFDs has been tested against the D100S1R5B (D series, 1PH AC 220V 50/60hz input, 8A output) inverter.
It should work for other VFDs in the D series too as well as the E series since its manual has been used for reference of the parameters and protocol details.
If controlling the VFD through RS485 isn't working for some reason both the D and E series support a 0-10V interface as well.
This includes controlling the direction through another input of the VFD. (See manual for details)
Here is a selection of the parameters needed to make the VFD (D and E series) communicate with FluidNC:
Selection of RS485 as control and frequency source:
Register | Value | Description | Possible Values |
---|---|---|---|
P0-000 | 2 | Command source | 0: Keypad 1: Control inputs 2: RS485 |
P0-001 | 0 | Frequency source | 0: main frequency source 1: auxiliary frequency source 2: main + aux 3: max(main, aux) 4: selectd by control input |
P0-002 | 6 | Main frequency source selection | 0: Keypad Potentiometer 1: Keypad Up Down Arrow 2: AIN1 3: AIN2 4: Multistep speed 5: PID 6: RS485 7: Internal PLC |
RS485 parameters:
Register | Value | Description | Possible Values |
---|---|---|---|
P0-055 | any free address between 1 and 31 | Address of VFD | 1- 31: slave addresses 2: master address |
P0-056 | 2 works just fine | Baudrate | 0: 2400bps 1: 4800bps 2: 9600bps 3: 19200bps 4: 38400bps |
P0-057 | 0 | Data framing | 0:1 start bit, 8 data bits, no parity, 1 stop bit 1: 1 start bit, 8 data bits, even parity, 1 stop bit 2: 1 start bit, 8 data bits, odd parity, 1 stop bit |
Setting min and max speed:
Register | Value | Description | Possible Values |
---|---|---|---|
P0-007 | Whatever your spindle can handle | Max frequency | Min frequency - 600hz |
P0-008 | Whatever your spindle can handle | Min frequency | 0 - Max frequency |
The following registers are read / written to by FluidNC:
Read access only:
Register | Description |
---|---|
0x007 | Max frequency in hz * 100, same as config parameter P0-007 |
0x008 | Min frequency in hz * 100, same as config parameter P0-008 |
0x300 | Current fault number 0 = no fault 1-18 = fault number |
0x500 | VFD status Bit 0: run, 1=run, 0=stop Bit 1: direction, 1=ccw, 0=cw Bit 2: control, 1=local, 0=remote Bit 3: sight fault, 1=fault, 0=no fault Bit 4: fault, 1=fault, 0=no fault Bit 5-15: reserved |
0x502 | Current output frequency in hz * 100 |
Write access only:
Register | Description |
---|---|
0x900 | VFD control Bit 0: run, 1=run, 0=stop Bit 1: direction, 1=ccw, 0=cw Bit 2: jog, 1=jog, 0=stop Bit 3: reset, 1=reset, 0=dont reset Bit 4-15: reserved |
0x901 | Speed to be set in hz * 100 |
Example YAML config:
uart1:
txd_pin: gpio.14
rxd_pin: gpio.15
rts_pin: gpio.13
baud: 9600
mode: 8N1
NowForever:
uart_num: 1
modbus_id: 1
tool_num: 0
speed_map: 0=0% 24000=100%
off_on_alarm: false
Contributed via this PR
Contributed via this PR
SiemensV20:
uart:
txd_pin: gpio.17
rxd_pin: gpio.16
rts_pin: gpio.4
baud: 9600
mode: 8E1
modbus_id: 1
tool_num: 0
speed_map: 0=0% 24000=100%
The M4 (spindle reverse on) command will only be accepted if a direction pin is assigned to an I/O pin.
pwm:
pwm_hz: 5000
direction_pin: NO_PIN
output_pin: gpio.14
enable_pin: NO_PIN
disable_with_s0: false
s0_with_disable: true
spinup_ms: 0
spindown_ms: 0
tool_num: 0
speed_map: 0=0.000% 10000=100.000%
off_on_alarm: false
This is like a PWM spindle except that you have separate PWM signals for clockwise (CW) and counterclockwise (CCW) rotation. This was specifically designed to directly control a H bridge circuit.
HBridge:
pwm_hz: 5000
output_cw_pin: gpio.4
output_ccw_pin: gpio.16
enable_pin: gpio.26
disable_with_s0: false
spinup_ms: 1000
spindown_ms: 1000
tool_num: 100
speed_map: 0=0.000% 10000=100.000%
off_on_alarm: false
A laser is considered a spindle because gcode does not have laser specific codes. It uses the RPM value as a power level. Lasers also have special requirements.
They always operate like the advanced laser mode of Grbl
speed_map: final xxx=100% can be whatever you want, but it is typically 255 or 1000. This would need to be used in the CAM software as the max power number.
off_on_alarm: true: is recommended from a safety point of view to ensure the laser is switched off when movement is stopped due to triggered alarm.
The 2 modes are quite different and each optimized for different types of work.
M3 Mode
This mode is primarily used for cutting through parts. The laser operates whenever you are in a feed rate controller mode (G1, G2 or G3). It will stay on at all times at the full Snnn value. This includes when there is no motion. To stop the laser you must send M5, G0 or S0. This gives you full control. For example, you may want to dwell a fraction of a second at the start or end of a cut.
Here is an example of a macro to test the laser at minimal power
M3 S1 ; lowest power
G1 F100 ; set G1 and an arbitray feedrate to turn on the laser
G4 P0.50 ; wait 0.5 seconds
G0 ; turn off the laser
M5 ; keep it off.
M4 Mode
M4 mode is primarily used for engraving. It compensates to lower the power of the laser during acceleration and deceleration to prevent darkening those sections. It will stay off when there is no motion.
Config example
Laser:
pwm_hz: 5000
output_pin: gpio.4
enable_pin: NO_PIN
disable_with_s0: false
s0_with_disable: true
tool_num: 0
speed_map: 0=0.000% 255=100.000%
off_on_alarm: true
This is like a PWM signal except that the pin will be full on for any speed above 0 that you select. PWM signals can quickly destroy a relay.
relay:
direction_pin: NO_PIN
output_pin: gpio.26
enable_pin: NO_PIN
disable_with_s0: false
s0_with_disable: true
spinup_ms: 0
spindown_ms: 0
tool_num: 0
speed_map: 0=0.000% 0=100.000% 1=100.000%
off_on_alarm: false
This is a default spindle that is automatically created if you did not specify a spindle in your config file.
NoSpindle:
You can define as many spindles as your hardware will support. They will act totally independently. You must use separate I/O pins for each one. Simply add each spindle definition to the config file.
Each must have a unique tool_num: If you have a gap in numbers, like the first spindle is tool_num: 0
and the second spindle is tool_num: 10
, numbers 0 through 9 would belong to the first spindle. This allows any spindle to be an ATC (automatic tool changing) spindle.
If you only have 1 spindle defined you should use tool_num: 0. The spindle will accept all tool numbers.
You switch between tools with the tool change gcode (M6). M6 T2 would switch to the spindle that covers tool number 2. If the spindles are offset in one or axis, you will need to deal with that yourself.
Ways to to rezero the new spindle.