-
-
Notifications
You must be signed in to change notification settings - Fork 39
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
24116d7
commit 2ee5e07
Showing
29 changed files
with
932 additions
and
111 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -156,4 +156,5 @@ frame_data/ | |
*.gif | ||
*.mp4 | ||
*.png | ||
*.avi | ||
*.avi | ||
*.pkl |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
ko_fi: furrysalamander |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,2 +1,99 @@ | ||
# rubedo | ||
|
||
# Rubedo | ||
## A tool for automatically calibrating pressure advance. | ||
Do you hate manually tuning pressure advance? Me too! This script can use a line laser and camera that are attached to your printer's toolhead to generated a 3D model of a pressure advance calibration pattern, and estimate how badly each line is deformed. We can pick the best one, and use this as our pressure advance value. | ||
|
||
data:image/s3,"s3://crabby-images/8adef/8adef71c39d6cf94d4e7495bb6ef6cd348b10f24" alt="Printer with the system installed" | ||
*The system installed on my printer* | ||
|
||
data:image/s3,"s3://crabby-images/39a8e/39a8e00b6f498b109af0a88f75d3d4a3e77f8f58" alt="Calibration pattern compared with the control and results" | ||
*On the left is the pattern that was scanned. The middle pattern shows the pattern printed with pressure advance disabled, and the pattern on the right shows a pattern printed with the calibrated value.* | ||
|
||
*** | ||
### Disclaimer | ||
This tool is still very much experimental. I hope that someday it will grow to be more robust and user friendly, but right now you should know at least a little bit of Python if you're interested in trying it out. It's not perfect. The gcode that is generated may not play nicely with your printer. I would recommend keeping the emergency stop button within close reach the first time you run the calibration, just to be safe. | ||
*** | ||
|
||
## Things you will need to setup on your printer: | ||
* Klipper + Moonraker. This project communicates with moonraker using websockets. | ||
* A high quality USB camera. I used the 1080p nozzle camera from 3DO. The exact model I have is no longer available for purchase, [but they do have a 4k version](https://3do.dk/59-dyse-kamera). You will need to adjust several of the constants in `constants.py` folder to adjust the system to work with whatever camera you choose. The most important thing here is that the camera can focus on things at close distances. I think it might be possible to get a modified endoscope working with the system, but I tried one without modifications and I just couldn't get enough detail to make this work well at the default focus distance. | ||
* A line laser, preferably with an adjustable focus. I tested a fixed focus laser earlier on, but the beam was super thick and there was no way to adjust it. I have a laser like [this](https://www.amazon.com/module-Industrial-Module-adjustable-point-2packs/dp/B0BX6Q9FD8/). It's capable of focusing at a fairly close distance, though I had to put some hot glue on the lens to keep it from rotating over time since it's fairly loose at the distances I'm using it at. Also, the outer edges of the line are poorly focused because they're further away than the center of the line, but it hasn't been a huge issue since it's just important that the very center of the beam is very thin. | ||
* `LASER_ON` and `LASER_OFF` macros. I have my laser connected to the SB2040 PCB on my toolhead. I used one of the 5v fan outputs, and it works fairly well. | ||
* Some sort of a mount for your USB camera and laser. I am using the mount `3DO_Mount_v2.step` located in the `camera_mounts` directory. If you do not use this exact mount, you will need to adjust the constants for the X and Y camera offset. In addition, you will need to ensure that the laser is at a 45° angle from the camera. Ideally, both the camera and laser will be most in focus exactly where the laser passes through the center of the camera's field of view. This was not the case with my hardware, so I have shift the area my code analyzes to the side a bit. If your camera is rotated differently than mine, it should be fairly easy to add another parameter to rotate the video feed, but I have not implemented that yet so you will need to figure that out yourself. There might be a way to do that with the ffmpeg arguments used. | ||
|
||
Here's a closeup of the system attached to my printer. I didn't have screws that were the right length, so I'm using some friction fit printed bushings and double sided tape to keep things together. | ||
|
||
data:image/s3,"s3://crabby-images/0ad87/0ad87e549711d839ecb419a48e695e45565ac888" alt="toolhead with system attached" | ||
|
||
## On whatever device you want to run the calibration from: | ||
* Install ffmpeg | ||
* Install the following python modules: | ||
``` | ||
opencv-python-headless matplotlib aiohttp websocket-client | ||
``` | ||
* Clone this repository | ||
* Go through the `constants.py` file, and adjust whatever you need to in order to make it match the way your printer is configured. | ||
### Configuring constants | ||
Honestly, this is going to be a bit tricky, and is probably the most brittle part of the process. Some of the config options, such as the camera X and Y offset are fairly straightforward. However, the crop settings are crucial to getting the system to work well. These are used by the `crop_frame()` function in `processing.py`, and if the resulting frame does not clearly show the area where the laser is hitting the filament, then this will not work at all. | ||
This image is what a cropped frame looks like on my machine. Perhaps I should shift the frame slightly further to the right. More experimentation needs to be performed to figure out exactly what the ideal setting is. The most important thing is that the camera should only see the line that it is currently scanning. You will need to take care to ensure that the line you're cropping to is not the wrong one. | ||
<img src="graphics/cropped.png" width="135" height="160"> | ||
``` | ||
TODO Add a simple script for calibrating this setting | ||
``` | ||
Assuming you are using an adjustable focus line laser, you will need to move the print head to the height where the print bed is perfectly in focus. Then, you will need to adjust the focus of the line laser until the laser beam is as thin as you can get it. Make sure that the beam is also perfectly parallel with the Y axis. This is necessary in order to get a good scan data. | ||
# Code Organization | ||
The code that allows for hands free calibration is in `main.py`. Once you've configured everything correctly, you should be able to run the script and get recommended pressure advance value. If you have `VALIDATE_RESULTS` enabled, the printer will print another two patterns, one w/ PA disabled, and another with the selected value. Most users probably won't want this, so feel free to turn it off. It also makes it hard to find the recommended value in the scripts output. | ||
If you want to test a different range of pressure advance values, or change where the calibration pattern is printed, you can do so by editing the parameters used for the PatternInfo object created at the start of the main function. By default, the pattern is printed at (30, 30), and tests 10 PA values between 0 and 0.06. | ||
### `pa.py` | ||
This file contains code for generating the calibration pattern gcode. | ||
### `record.py` | ||
This file contains functions for recording video clips of the printed lines. | ||
### `processing.py` | ||
This file contains a few utilities that filter and transform the video frames to help minimize noise. | ||
### `analysis.py` | ||
This file contains the code that generates height maps for each line, and then computes the line deviation for each heightmap. | ||
### `visualization.py` | ||
This file contains several utilities for visualizing the height maps and data that is generated from the scans. Currently, some of the functions here are in disrepair, as I needed the output during early development, but haven't used them since. | ||
### `generate_bulk_scans.py` | ||
This file will print 27 copies of the calibration pattern, scan all of them, and then save the results for later analysis. I used this while working on my research paper for this project to measure how consistent the system was. | ||
### `generate_report_data.py` | ||
This script consumes the files generated by the `generate_bulk_scans.py` script, and generates height map visualizations for each line that was scanned. In addition, it generates charts that aggregate the data from all scans to visualize the overal consistency of the calibration process. | ||
# Visualization | ||
Here are some heightmaps from lines that I scanned while working on this project. They're a little bit squished together because the columns actually represent video frames, and not pixels or millimeters. | ||
PA at 0.13 | ||
data:image/s3,"s3://crabby-images/0e609/0e609057d4dc99760abc61ede8aba79746bdd36a" alt="0.13 line" | ||
PA at 0.33 | ||
data:image/s3,"s3://crabby-images/f0887/f0887ad77757a20be9956841fbf20ffadfbc4fce" alt="0.33 line" | ||
PA at 0.60 | ||
data:image/s3,"s3://crabby-images/dbd22/dbd226e7d2035954098808ef19423479a298f97f" alt="0.60 line" | ||
Here's what the 0.33 line looks like in 3D | ||
data:image/s3,"s3://crabby-images/37d04/37d04315f4fb104e5228d2c82f287db082d1ad3a" alt="0.33 line 3D" | ||
# Results | ||
I am working on a formal research paper that goes more in depth about the way the system works, but in the meantime, I have a video on youtube that gives an overview of everything. | ||
# Additional Information | ||
If you think this project is cool, there are a bunch of us on the Alchemical3D discord server that are working on a printer that will use this system. In addition, we are excited about the prospect of using it to calibrate other things, such as extrusion multiplier or even bed mesh. Feel free to check it out: | ||
[https://discord.gg/ByyEByP7hp](https://discord.gg/ByyEByP7hp) | ||
# Support This Project | ||
Like this project? Feel free to make a donation. | ||
[data:image/s3,"s3://crabby-images/9535c/9535c379726b326df085d14ec0e6a32e614934c3" alt="ko-fi"](https://ko-fi.com/L3L63ISSH) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
/home/mike/Downloads/3DO_Mount_v2.step |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,8 +1,82 @@ | ||
OUTPUT_GRAPH = False | ||
OUTPUT_FRAMES = True | ||
OUTPUT_HEIGHT_MAPS = False | ||
# Connection details for communicating with the printer's moonraker API. | ||
HOST = '192.168.1.114' | ||
WS_PORT = 7125 | ||
GCODE_ENDPOINT = '/printer/gcode/script' | ||
OBJECTS_ENDPOINT = '/printer/objects/query' | ||
|
||
# This will print a calibrated + control pattern and measure the % improvement after tuning | ||
VALIDATE_RESULTS = True | ||
|
||
# Print settings | ||
BUILD_PLATE_TEMPERATURE = 110 | ||
HOTEND_TEMPERATURE = 255 | ||
HOTEND_IDLE_TEMP = 200 | ||
|
||
# This is where the toolhead moves to indicate that it's done printing the PA pattern. | ||
FINISHED_X = 30 | ||
FINISHED_Y = 250 | ||
|
||
# Any gcode you want to be sent before the pattern is printed. | ||
# You could just have this call PRINT_START if you've configured | ||
# that for your printer. | ||
PRINT_START = f""" | ||
M104 S180; preheat nozzle while waiting for build plate to get to temp | ||
M140 S{BUILD_PLATE_TEMPERATURE}; | ||
G28 | ||
M190 S{BUILD_PLATE_TEMPERATURE}; | ||
QUAD_GANTRY_LEVEL | ||
CLEAN_NOZZLE | ||
G28 Z | ||
M109 S{HOTEND_TEMPERATURE}; | ||
CLEAN_NOZZLE | ||
""" | ||
|
||
# Information about the USB camera mounted to the hotend. | ||
VIDEO_DEVICE = "/dev/video2" | ||
VIDEO_RESOLUTION = "1280x720" | ||
FRAMERATE = "30" | ||
# The camera's distance from the nozzle. | ||
# This tells the recording code how to center the line within the camera's field of view. | ||
# The offsets are in mm. | ||
CAMERA_OFFSET_X = 28 | ||
CAMERA_OFFSET_Y = 50.2 | ||
|
||
# This is the height where the camera and laser are in focus. | ||
LASER_FOCUS_HEIGHT = 17.86 | ||
|
||
# How the processing code finds the area of interest. Units are in pixels. | ||
# The crop offsets specify the pixel that the box should be centered on. | ||
CROP_X_OFFSET = 220 | ||
# In my case, the crop Y offset should be zero, but my offset Y value above is slightly off. | ||
# You can kind of tweak these if you find that things aren't quite right. | ||
CROP_Y_OFFSET = 11 | ||
# How big the area around the laser should be cropped to. | ||
CROP_FRAME_SIZE_X = 45 | ||
CROP_FRAME_SIZE_Y = 60 | ||
|
||
# Sometimes ffmpeg is slow to close. If we start moving too early, | ||
# we might accidentally record stuff we don't want to. | ||
# I would like to eliminate these eventually by improving the video recording code. | ||
FFMPEG_START_DELAY = 0.5 | ||
FFMPEG_STOP_DELAY = 0.6 | ||
|
||
# Pressure Advance Pattern Configuration | ||
# This changes how the gcode for the pressure advance pattern is generated. | ||
# Only edit this if you need to. | ||
Z_HOP_HEIGHT = 0.75 | ||
LAYER_HEIGHT = 0.25 | ||
RETRACTION_DISTANCE = 0.5 | ||
EXTRUSION_DISTANCE_PER_MM = 0.045899 | ||
BOUNDING_BOX_LINE_WIDTH = 0.4 # May need adjustment. | ||
|
||
# TODO: implement support for these. | ||
# If we know the FOV, we can attach actual units | ||
# to the values that are calculated. | ||
# CAMERA_FOV_X = 0 | ||
# CAMERA_FOV_Y = 0 | ||
|
||
# These were used earlier on in development, but I need to re-implement | ||
# them, as the refactor I did removed the code that made them work. | ||
# OUTPUT_GRAPH = False | ||
# OUTPUT_FRAMES = True | ||
# OUTPUT_HEIGHT_MAPS = False |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,2 @@ | ||
!*.png | ||
!*.jpg |
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.