DIY Filament Runout Sensor for 3D Printer Fault Detection using Arduino

Published  March 4, 2022   0
DIY 3D Printer Filament Runout Sensor

3D printers have become an essential aspect of quick prototyping and have reached out to a wide range of makers and engineers. But most of these printers are hobby grade and hence require constant tinkering and repairs to keep them working. Among all challenges, 3D printer filament breaking or tangling is one of the most common problems that everyone who uses a 3D printer has encountered at some point, as a lot of print and filament gets wasted because of this issue. To avoid this issue, we'll use Filament Sensors, which will allow your printer to recognize when it's out of filament and stop printing while you change spools. 

Today I'll demonstrate how to create a 3D printed smart filament sensor on perf board using an optical encoder sensor and a controller. I designed a separate sensor housing for our filament sensor since I wanted to utilize it on other printers as well. When the filament stops moving due to breakage or tangling, this Filament Sensor will beep to alert you.

You can check out the demonstration video given below-

Material Required to build a DIY Filament Runout Sensor

Filament Sensor Components

Optical Encoder Sensor

For Sensing mechanism part:

  1. BB3609 Optical encoder sensor X 1
  2. Bearing 625 X 3
  3. M5 30mm Allen Bolt with Nut X 1
  4. PC4 M5 Pneumatic Coupler X 2
  5. Spring X 1

For Controller and buzzer part:

  1. AT-Tiny 85 X 1
  2. DPDT Push to on Switch
  3. 2 Pin wire Connector
  4. 7805
  5. 7812
  6. Active Buzzer 5V
  7. Female Bergstrip

How Filament Sensor Works

Since there are two types of filament sensors, one detects the presence of filament using a limit switch and the other uses optical rotary encoders to feel the velocity of the filament moving within. Each filament sensor has its own set of advantages and disadvantages.

The one with the limit switch is for filament runover or when the extruder motor tends to run dry due to filament breaking. It is, nevertheless, simple to set up and maintain because of its simplicity and absence of moving components, and its compactness makes it more useful. However, it will not be engaged if the filament breaks near the sensor's output since the filament's presence will still be there after the filament breakup.

3D Printer Failure Detection Sensor

Fig1: In the above image, you can see despite the fact that the filament is broken at the output, the switch still indicates the filament's presence

When it comes to the optical rotary-based sensor, it will notify you regardless of the location and extent of the filament breakage. When the filament tangling causes the filament to halt, it will beep. Now, if the filament breaks anywhere and is unable to reach the extruder motor in any manner, the sensor will beep, indicating that something went wrong.

Because we are more concerned with precision and effectiveness, we will build our filament sensor using an optical rotary sensor MOC7811-based module, which will ensure that the filament moves continuously.

Here you can view the current Sensors that are available on the market.

Limit Switch based Filament Sensor

Fig 2: Limit Switch based Filament Sensor that can detect the presence of the filament within

Rotary Encoder Sensor

Fig 3: The above image shows a Rotary encoder based sensor that senses the presence as well as the movement of the filament

Motion-based Filament Sensor

Since we've spoken about the benefits and drawbacks of the limit switch sensor. Now let's talk about the filament motion-based Sensors.

These sensors are Smart Filament Sensors, which are incredibly accurate and useful in detecting filament clogging, tangling, and breaking, among other things.

The sensor we'll build will allow us to modify the time delay or inactivity of the filament movement, which will be determined by the printing quality and speed. If the filament traversal speed is sluggish, the printing speed is also slow. For personalization, we'll create a rotor that will be mounted to the potentiometer's knob. It will also contain a deactivate sensor button, which will turn off the sensor when it is not in use.

3D Printer Filament Runout Sensor Circuit Diagram

You can see the final connection in the below image.

DIY Filament Detection Sensor

I have used a ATtiny 85 digispark board for detecting the rotation and controlling the buzzer.

Programming AT-Tiny 85

I used Arduino IDE to program the ATtiny85 digispark board. If you're new to Digispark Boards and want to learn how to program ATtiny85 with the Arduino IDE, click the link.

We have previously made many exciting projects using this ATtiny85 controller, you can check that out.

The following program is written such that every time the rotary wheel moves with filament, an interrupt is triggered.

#include <avr/io.h>
#include <avr/interrupt.h>
// Define pins for switch the LED, plus the chosen interrupt for reacting to
#define INTERRUPT_PIN PCINT1  // This is PB1 per the schematic
#define INT_PIN PB1 // ON Board LED Interrupt pin of choice: PB1 (same as PCINT1) - Pin 6
#define LED_PIN PB4 // PB4 - Pin 3
#define BUZZ PB0
#define PCINT_VECTOR PCINT0_vect // This step is not necessary - it's a naming thing for clarity
// FYI: Variables used within ISR must be declared Volatile.
// static volatile byte LEDState; 
// The setup function runs only once when the ALU boots
void setup() {
    pinMode(LED_PIN, OUTPUT);  // Set our chosen LED visual feedback as an output pin (PB4 / Pin 3)
    pinMode(BUZZ, OUTPUT);    
    digitalWrite(LED_PIN, HIGH); // Blink to show it's booting up and your LED is wired correctly
    delay(500);
    digitalWrite(LED_PIN, LOW);
    delay(500);
    // Code here is the key piece of configuring and enabling the interrupt 
    cli(); // Disable interrupts during setup
    PCMSK |= (1 << INTERRUPT_PIN); // Enable interrupt handler (ISR) for our chosen interrupt pin (PCINT1/PB1/pin 6)
    GIMSK |= (1 << PCIE);  // Enable PCINT interrupt in the general interrupt mask
    pinMode(INT_PIN, INPUT_PULLUP);   // Set our interrupt pin as input with a pullup to keep it stable
    sei();  //last line of setup - enable interrupts after setup
}
unsigned long current_millis=0;
unsigned long prev_millis=0;
int interval;
void loop() {
  // Put any general processing code in the main loop
  // Be aware - if the INT_PIN changes state, the loop will be suspended while the ISR runs to completion
  interval=map(analogRead(1),0,1023,0,20);
  current_millis=millis();
  if ((unsigned long)(current_millis-prev_millis)>=interval*1000)
  {
    digitalWrite(BUZZ,1);
    delay(50);
    digitalWrite(BUZZ,0);
    delay(50);
    digitalWrite(BUZZ,1);
    delay(50);
    digitalWrite(BUZZ,0);
    delay(50);
    digitalWrite(BUZZ,1);
    delay(50);
    digitalWrite(BUZZ,0);
    delay(50);
    digitalWrite(BUZZ,1);
    delay(50);
    digitalWrite(BUZZ,0);
    delay(150);
   }
}
// This is the interrupt handler called when there is any change on the INT_PIN
// ISR is defined in the headers - the ATtiny85 only has one handler
ISR(PCINT_VECTOR)
{ prev_millis=millis();
  if( digitalRead(INT_PIN) == HIGH ) {
    digitalWrite(LED_PIN, HIGH);
  }else{
    digitalWrite(LED_PIN, LOW);
  }
}

3D Design of the Enclosure

I got the idea from Thingiverse. There, I came across a design that was exactly what I needed. Unfortunately, I was unable to locate all of the needed components included in that design's BOM, so I had to modify the design, which you may obtain from the provided links. The controller circuit needed to be housed in its own add-on box now.

Here you can find the images of the design.

Filament Runout Sensor 3D Enclosure

I have changed the spring holder and other attachments for the circuit we have made. Modified Design is as follows.

Filament Runout Sensor 3D Design

And this is the final outlook of our 3D Printer Filament Runout Sensor.

DIY 3D Printer Filament Runout Sensor

Conclusion

Hence we have made our own smart filament Sensor using an optical encoder sensor and AT tiny 85 Controller. I hope you have enjoyed the project. For any queries please refer to our forum.

Complete Project Code

#include <avr/io.h>
#include <avr/interrupt.h>
// Define pins for switch the LED, plus the chosen interrupt for reacting to
#define INTERRUPT_PIN PCINT1  // This is PB1 per the schematic
#define INT_PIN PB1           // ON Board LED Interrupt pin of choice: PB1 (same as PCINT1) - Pin 6
#define LED_PIN PB4           // PB4 - Pin 3
#define BUZZ PB0
#define PCINT_VECTOR PCINT0_vect      // This step is not necessary - it's a naming thing for clarity
// FYI: Variables used within ISR must be declared Volatile.
// static volatile byte LEDState;
// The setup function runs only once when the ALU boots
void setup() {
    pinMode(LED_PIN, OUTPUT);         // Set our chosen LED visual feedback as an output pin (PB4 / Pin 3)
    pinMode(BUZZ, OUTPUT);   
    digitalWrite(LED_PIN, HIGH);      // Blink to show it's booting up and your LED is wired correctly
    delay(500);
    digitalWrite(LED_PIN, LOW);
    delay(500);
    // Code here is the key piece of configuring and enabling the interrupt
    cli();                            // Disable interrupts during setup
    PCMSK |= (1 << INTERRUPT_PIN);    // Enable interrupt handler (ISR) for our chosen interrupt pin (PCINT1/PB1/pin 6)
    GIMSK |= (1 << PCIE);             // Enable PCINT interrupt in the general interrupt mask
    pinMode(INT_PIN, INPUT_PULLUP);   // Set our interrupt pin as input with a pullup to keep it stable
    sei();                            //last line of setup - enable interrupts after setup
}
unsigned long current_millis=0;
unsigned long prev_millis=0;
int interval;
void loop() {
  // Put any general processing code in the main loop
  // Be aware - if the INT_PIN changes state, the loop will be suspended while the ISR runs to completion
  interval=map(analogRead(1),0,1023,0,20);
  current_millis=millis();
  if ((unsigned long)(current_millis-prev_millis)>=interval*1000)
  {
    digitalWrite(BUZZ,1);
    delay(50);
    digitalWrite(BUZZ,0);
    delay(50);
    digitalWrite(BUZZ,1);
    delay(50);
    digitalWrite(BUZZ,0);
    delay(50);
    digitalWrite(BUZZ,1);
    delay(50);
    digitalWrite(BUZZ,0);
    delay(50);
    digitalWrite(BUZZ,1);
    delay(50);
    digitalWrite(BUZZ,0);
    delay(150);
   }
}
// This is the interrupt handler called when there is any change on the INT_PIN
// ISR is defined in the headers - the ATtiny85 only has one handler
ISR(PCINT_VECTOR)
{ prev_millis=millis();
  if( digitalRead(INT_PIN) == HIGH ) {
    digitalWrite(LED_PIN, HIGH);
  }else{
    digitalWrite(LED_PIN, LOW);
  }
}
Video

Have any question realated to this Article?

Ask Our Community Members