最新消息: USBMI致力于为网友们分享Windows、安卓、IOS等主流手机系统相关的资讯以及评测、同时提供相关教程、应用、软件下载等服务。

Delta (rostock型)3d打印机算法解读及调试步骤

IT圈 admin 42浏览 0评论

2024年7月22日发(作者:虢痴旋)

Delta (rostock型)3d打印机算法解读及调试步骤

一、

前言

Delta机型是一种并联式运动结构的3d打印机,delta机型实际上是分为两大类,一

种是工业上用的并联式机器人;另外一种是rostock 的运动结构。这两种结构做出来的打

印机给人的感觉都是非常cool的。所以大伙都偏好这类型的打印机,目前主流的固件

marlin 和repetier都支持rostock结构的3d打印机。这里鸭哥就谈谈自己对rostock机

型的运动算法的理解,继而从运动算法中推算一下如何调试一台rostock的参数。

二、

基础知识

要理解rostock的全部运动算法所要涉及的数学知识不是太多,如果不记得的话,是

时候找高中数学老师喝喝茶吃个小饭啦。

1)三角函数 sin cos 这个是理解rostock计算过程的基础知识

2)笛卡尔空间坐标转换/线性代数,这个嘛是属于进阶内容,如果懂那就更好啦,如果

不懂也没关系,在把所有的外界条件全部设置为理想情况下,这个笛卡尔空间坐标转换可

以不用精通的。(包括鸭哥也不敢说对笛卡尔空间坐标转换和线性代数精通哦)

3)marlin程序的结构逻辑结构(对于arduino ide /arduino程序基本教程,这里就不

展开了,不懂的小白们可以先学一下arduino的基础教程先,饭要一口一口得出,路要一

步一步的走)

三、

Marlin程序解读

这里鸭哥不打算讲marlin的整个loop()函数的流程,讲讲delta机型的核心部分。

对于marlin来说,delta机型和非delta机型在对于温控、看门狗、电机运动甚至空间坐

标等方面都是一样的。区别在哪里呢?区别就在与delta多了一个笛卡尔坐标转换的函数

Marlin的loop()主体流程

Void loop ()

{

Get_command() ; //从sd卡或者串口获取gcode

Process_command(); //解析gcode并且执行代码

Manage_heater();//控制机器的喷头和热床的温度

Manage_inactivity();//

checkHitEndstops();//检查endstop的状态

Lcd_update(); //更新lcd 上面的信息

}

在这个过程中 process_command()是控制的核心,各位仔细研读一下

process_command()的代码就发现arduino的厉害了。简单说一下process_command()

的流程,说白了,process_command()就是一个巨大的case 结构,这里讲讲G1命令的大

致逻辑(G1命令不知道的自己搜索去):

Process_command()

{

Case 0: //g0->g1

Case 1 :

{

if(Stopped == false) {

get_coordinates(); // 获取当前的坐标,这里是指打印件的世界坐标哦,不是

delta的xyz电机的坐标哦!普通结构的打印机则是一样的。

#ifdef FWRETRACT

if(autoretract_enabled)

if( !(code_seen('X') || code_seen('Y') || code_seen('Z')) &&

code_seen('E')) { //获取 命令中 xyze轴的参数

Float echange=destination[E_AXIS]-current_position[E_AXIS]; //这里是

算最小回抽值的,如果移动距离小于最小回抽值就不回抽了。这里是一个辅助功能。简单

了解可以了。

if((echange<-MIN_RETRACT && !retracted) ||

(echange>MIN_RETRACT && retracted)) { //move appears to be an attempt to

retract or recover

current_position[E_AXIS] = destination[E_AXIS]; //hide the

slicer-generated retract/recover from calculations

plan_set_e_position(current_position[E_AXIS]); //AND from the

planner

retract(!retracted);

return;

}

}

#endif //FWRETRACT

prepare_move(); //执行移动命令

return;

}

}

从上面的代码来看呢,对于运动类的Gcode,marlin会在process_command()函数

中获取xyze各轴的参数后算出目标坐标(destination[_AXIS]),也会使用

get_coordinates()来获取当前坐标(current_position[E_AXIS])(再次强调,这个坐标是

打印件的世界坐标),当我们知道了目标坐标和当前坐标以后,空间中移动的距离就可以算

出来了(不会算的,请自觉请高中数学老师吃饭去),接下来marlin就使用perpare_move()

来控制电机啦。

接下来呢很自然就要讲讲prepare_move()这个函数啦。先上代码先,代码鸭哥做了精

简,只看关键的部分就是delta和普通结构的代码,先说一下plan_buffer_line()这个函

数的作用的把坐标数组current_position[i] 、 destination[i] 放到一个内存的一个缓存

区里面,然后控制电机转多少圈这样一个作用的,具体代码可以自己去看,在一旦进入这

个函数以后,delta和普通机型的代码都是一样的,也就是说delta和普通结构的电机控制

其实是一样的。

Difference[i]数组 :用来储存目标坐标和当前坐标之间的距离的,(这里是包含了xyze

轴的数组)

Destination[i]数组:目标坐标的数值,是从process_command()函数中G1读取

XYZE参数获取的。

Current_position[i]数组:当前坐标的数值,是从G1 命令中get_coordinates()传递

过来的。如果是3个轴都归零的情况下,current_position就是储存三个坐标原点,如果

开始运动了,这里的值就是上一个prepare_move()循环执行后上一次的destination[i]的

值。(这个下面会有看到赋值语句)

Delta[i]数组:delta打印机的xyz三个电机要移动的距离

void prepare_move()

{

#ifdef DELTA // 设置机子是delta机型(rostock)

float difference[NUM_AXIS]; //定义目标距离,用于转换坐标用的过渡变量

for (int8_t i=0; i < NUM_AXIS; i++) {

difference[i] = destination[i] - current_position[i];

} //计算世界坐标的距离值

//***开始计算笛卡尔距离 并且暴力直线插值来减少运算量***//

float cartesian_mm = sqrt(sq(difference[X_AXIS]) +

sq(difference[Y_AXIS]) +

sq(difference[Z_AXIS]));

if (cartesian_mm < 0.000001) { cartesian_mm = abs(difference[E_AXIS]); }

if (cartesian_mm < 0.000001) { return; }

float seconds = 6000 * cartesian_mm / feedrate / feedmultiply;

int steps = max(1, int(delta_segments_per_second * seconds));

for (int s = 1; s <= steps; s++) {

float fraction = float(s) / float(steps);//直线插值

for(int8_t i=0; i < NUM_AXIS; i++) {

destination[i] = current_position[i] + difference[i] * fraction;

}

//***结束计算笛卡尔距离 并且暴力直线插值来减少运算量***//

calculate_delta(destination);//将打印件的世界坐标转换为xyz电机轴的运动量

plan_buffer_line(delta[X_AXIS], delta[Y_AXIS], delta[Z_AXIS],

destination[E_AXIS], feedrate*feedmultiply/60/100.0,

active_extruder);

}

#endif // DELTA

。。。。。。。。。。。。

#if ! (defined DELTA || defined SCARA)

// Do not use feedmultiply for E or Z only moves

if( (current_position[X_AXIS] == destination [X_AXIS]) &&

(current_position[Y_AXIS] == destination [Y_AXIS])) {

plan_buffer_line(destination[X_AXIS], destination[Y_AXIS],

destination[Z_AXIS], destination[E_AXIS], feedrate/60, active_extruder); //直接将

destination[i]的值发送去运动缓存里面

}

else {

plan_buffer_line(destination[X_AXIS], destination[Y_AXIS], destination[Z_AXIS],

destination[E_AXIS], feedrate*feedmultiply/60/100.0, active_extruder);

}

#endif // !(DELTA || SCARA)

for(int8_t i=0; i < NUM_AXIS; i++) {

current_position[i] = destination[i]; //更新当前坐标的值为刚执行的目标坐标值

}

}

好,看了一大段代码后小结一下。对于普通结构来说,G1 每次将新读取gcode代码

参数传递给prepare_move()函数中destination[i]数组以后,prepare_move()就会将其

传递到plan_buffer_line()进行电机的运动。而delta结构呢,就相对复杂一点,G1命

令读取了gcode代码参数后也是传递到prepare_move()函数中destination[i],然后

marlin要计算目标坐标与当前坐标的笛卡尔距离,然后通过固定时间间隔的方式来将笛卡

尔距离分成若干个小直线,通过这样的方式来就减少cpu的浮点预算量,然后再通过

calculate_delta[i]函数来将简化后的destination[i]换算成三个电机的运动坐标,并传递到

delta[i]中,接下来就是plan_buffer_line()了。

最后!到了最后了!来看看calculate_delta()函数,这个函数的主要用途是将打印

件的世界坐标转换为三个垂直的电机轴的运动坐标哦。注意:新的marlin支持SCARA结

构的delta,那里也有个calculate_delta()的函数,不过那个跟rostock有点差异。所

以我们还是看rostock的吧。

void calculate_delta(float cartesian[3])

{

delta[X_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower1_x-cartesian[X_AXIS])

- sq(delta_tower1_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

delta[Y_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower2_x-cartesian[X_AXIS])

- sq(delta_tower2_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

delta[Z_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower3_x-cartesian[X_AXIS])

- sq(delta_tower3_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

/*

SERIAL_ECHOPGM("cartesian x="); SERIAL_ECHO(cartesian[X_AXIS]);

SERIAL_ECHOPGM(" y="); SERIAL_ECHO(cartesian[Y_AXIS]);

SERIAL_ECHOPGM(" z="); SERIAL_ECHOLN(cartesian[Z_AXIS]);

SERIAL_ECHOPGM("delta x="); SERIAL_ECHO(delta[X_AXIS]);

SERIAL_ECHOPGM(" y="); SERIAL_ECHO(delta[Y_AXIS]);

SERIAL_ECHOPGM(" z="); SERIAL_ECHOLN(delta[Z_AXIS]);

*/

}

代码很简单delta[i]是指电机轴的运动坐标,cartesian[i]是指打印件的世界坐标,从

上面的程序来看就是从prepare_move()中经过插值简化的destination[i]。大伙随便看

一个轴的换算

delta[X_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower1_x-cartesian[X_AXIS])

- sq(delta_tower1_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

delta_diagonal_rod_2 是推杆长的平方

delta_tower1_x 是左前柱的x坐标值,是由radius这个参数算出来的

delta_tower1_y 是左前柱的y坐标值,是由radius这个参数算出来的

具体怎么算就看下面这个函数

void recalc_delta_settings(float radius, float diagonal_rod)

{

delta_tower1_x= -SIN_60*radius; // front left tower

delta_tower1_y= -COS_60*radius;

delta_tower2_x= SIN_60*radius; // front right tower

delta_tower2_y= -COS_60*radius;

delta_tower3_x= 0.0; // back middle tower

delta_tower3_y= radius;

delta_diagonal_rod_2= sq(diagonal_rod);

}

好了回顾一下marlin的delta机型参数是需要什么?

推杆的长度、电机轴上滑块的宽度、喷头支架的宽度,还有三个电机的圆半径。对不

对?忘了?!不要紧,给你看看代码

//================================================

=================

//========================Delta

=============================

Settings

//================================================

=================

// Enable DELTA kinematics and most of the default configuration for Deltas

#define DELTA

// Make delta curves from many straight lines (linear interpolation).

// This is a trade-off between visible corners (not enough segments)

// and processor overload (too many expensive sqrt calls).

#define DELTA_SEGMENTS_PER_SECOND 200

// NOTE NB all values for DELTA_* values MUST be floating point, so always

have a decimal point in them

// Center-to-center distance of the holes in the diagonal push rods.

#define DELTA_DIAGONAL_ROD 250.0 // mm //杆长

// Horizontal offset from middle of printer to smooth rod center.

#define DELTA_SMOOTH_ROD_OFFSET 175.0 // mm //电机轴的圆半径

// Horizontal offset of the universal joints on the end effector.

#define DELTA_EFFECTOR_OFFSET 33.0 // mm // 装喷嘴的平台的中心到杆连接

处的距离

// Horizontal offset of the universal joints on the carriages.

#define DELTA_CARRIAGE_OFFSET 18.0 // mm //电机轴滑块的距离

// Effective horizontal distance bridged by diagonal push rods.

#define DELTA_RADIUS

(DELTA_SMOOTH_ROD_OFFSET-DELTA_EFFECTOR_OFFSET-DELTA_CARRIAGE_OFF

SET)

通过上述的参数可以算出一个DELTA_RADIUS ,这个delta_radius就是上面

“delta_tower1_x 是左前柱的x坐标值,是由radius这个参数算出来的 ”里面的radius

了。

至此所有有关与delta的运动的代码已经通读了一遍。下面就开始分析分析代码和运

动的关系了。

四、

Rostock运动分析

下面是Rostock的结构示意图,分析的第一步是简化整个结构,这里就需要将XY电

机的两个竖轴投影到Z轴的平面上,下图中红色线框画出来的就是z轴的平面,同时我们

可以不考虑XY电机的推杆的运动情况,因为可以XY电机轴的运动可以通过投影在z轴平

面上的虚拟轴笛卡尔空间变换转换回去的。

投影好了以后接下来把z轴放平,那么单独考虑z轴情况,这个情况是在坐标原点的

z电机轴与推杆的情况。为了简化过程,鸭哥已经把z轴滑块,喷头平台都设定为0 了。

那么,z电机轴方向便形成了一个三角形,推杆、radius和z轴电机上的电机坐标,这个

时候三角函数出来啦! 推杆

2

= radius

2

+ 电机坐标

2

在这三角形中推杆是不变的,另外

三角形始终都会是一个直角三角形。一定要记住这几个条件哦。

好,现在我们假设喷头只在x轴上运动,z轴y轴都不动。如果打印件的世界坐标移

动dX距离,rostock需要考虑的问题就是怎么讲dX转换为z轴电机的移动距离了。下图

就是用来表示这种情况。由于推杆长度是不变的,那么

推杆

2

= radius

2

+ 电机坐标

2

就变成下面这样

推杆

2

= (radius +dX)

2

+ (电机坐标-dx)

2

Radius的距离换成destination[x],电机坐标换成delta[x]

推杆

2

= (destination[x +dX])

2

+ (delta[x-dx])

2

Marlin中calculate_delta()这个函数其实就是算

推杆

2

= (destination[x +dX])

2

+ (delta[x-dx])

2

这个等式明白啦,打印件X轴和Y轴的运动分析就明白啦。

另外,再看看打印件z轴的运动分析,还是看看源代码

delta[X_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower1_x-cartesian[X_AXIS])

- sq(delta_tower1_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

Cartesian[z]是没有在sqrt函数里面的,而是直接加在delta[x]的值上面的。

所以,在调机的时候应该先调z轴,这里就是原因。因为在XYZ三个轴的坐标中只有

Z轴是直接通过同步轮和电机脉冲就可以调准的。调准了z轴以后再调XY轴才是对的。

五、

调机心得

下面是鸭哥的调机的一些心得,与大家分享一下。首先是调机的顺序:

1)选择同步轮可以选择GT2 20齿/40齿 。因为GT2是2mm齿距,整数齿可以使

脉冲数为整数,同时也减少三角函数运算中浮点运算的压力,同时也简化自己的调整步骤。

2)调机时先调z轴运动方向的精度,这个是直接用同步轮和脉冲数就可以调好的。

3)z轴调好以后就调整x轴或者Y轴的,看看运动的路径是不是呈现平面状态,如果

喷头运动路径是弧线这个时候就要调整radius了,增加或者减少radius的值来调整运动

路径是一个平面

4)完成上述步骤以后就可以试打了,这个时候就可以看看XY轴的打印误差是多少了。

如果杆长等硬件参数都比较准确的话那打印误差不会有多少的。如果XY轴有误差的话就

要根据误差大小来等量调整radius这个变量。对应代码是

// Horizontal offset from middle of printer to smooth rod center.

#define DELTA_SMOOTH_ROD_OFFSET 175.0 // mm //电机轴的圆半径

等量修改,比如X轴偏大0.1mm,那么 调整量就是175.0-0.1 = 174.9 了哦。这样

反复几次就基本调好了。

六、

后记

本文是鸭哥的一些心得体会,是个人对marlin还有rostock的一些了解,当然也许会

有这样那样的错误和不足,鸭哥非常欢迎大神高手批评指正。本文首发于珠海创客空间网

站,欢迎转载,著名出处。欢迎修改,修改的地方用另外的文字著名就可以了。

2024年7月22日发(作者:虢痴旋)

Delta (rostock型)3d打印机算法解读及调试步骤

一、

前言

Delta机型是一种并联式运动结构的3d打印机,delta机型实际上是分为两大类,一

种是工业上用的并联式机器人;另外一种是rostock 的运动结构。这两种结构做出来的打

印机给人的感觉都是非常cool的。所以大伙都偏好这类型的打印机,目前主流的固件

marlin 和repetier都支持rostock结构的3d打印机。这里鸭哥就谈谈自己对rostock机

型的运动算法的理解,继而从运动算法中推算一下如何调试一台rostock的参数。

二、

基础知识

要理解rostock的全部运动算法所要涉及的数学知识不是太多,如果不记得的话,是

时候找高中数学老师喝喝茶吃个小饭啦。

1)三角函数 sin cos 这个是理解rostock计算过程的基础知识

2)笛卡尔空间坐标转换/线性代数,这个嘛是属于进阶内容,如果懂那就更好啦,如果

不懂也没关系,在把所有的外界条件全部设置为理想情况下,这个笛卡尔空间坐标转换可

以不用精通的。(包括鸭哥也不敢说对笛卡尔空间坐标转换和线性代数精通哦)

3)marlin程序的结构逻辑结构(对于arduino ide /arduino程序基本教程,这里就不

展开了,不懂的小白们可以先学一下arduino的基础教程先,饭要一口一口得出,路要一

步一步的走)

三、

Marlin程序解读

这里鸭哥不打算讲marlin的整个loop()函数的流程,讲讲delta机型的核心部分。

对于marlin来说,delta机型和非delta机型在对于温控、看门狗、电机运动甚至空间坐

标等方面都是一样的。区别在哪里呢?区别就在与delta多了一个笛卡尔坐标转换的函数

Marlin的loop()主体流程

Void loop ()

{

Get_command() ; //从sd卡或者串口获取gcode

Process_command(); //解析gcode并且执行代码

Manage_heater();//控制机器的喷头和热床的温度

Manage_inactivity();//

checkHitEndstops();//检查endstop的状态

Lcd_update(); //更新lcd 上面的信息

}

在这个过程中 process_command()是控制的核心,各位仔细研读一下

process_command()的代码就发现arduino的厉害了。简单说一下process_command()

的流程,说白了,process_command()就是一个巨大的case 结构,这里讲讲G1命令的大

致逻辑(G1命令不知道的自己搜索去):

Process_command()

{

Case 0: //g0->g1

Case 1 :

{

if(Stopped == false) {

get_coordinates(); // 获取当前的坐标,这里是指打印件的世界坐标哦,不是

delta的xyz电机的坐标哦!普通结构的打印机则是一样的。

#ifdef FWRETRACT

if(autoretract_enabled)

if( !(code_seen('X') || code_seen('Y') || code_seen('Z')) &&

code_seen('E')) { //获取 命令中 xyze轴的参数

Float echange=destination[E_AXIS]-current_position[E_AXIS]; //这里是

算最小回抽值的,如果移动距离小于最小回抽值就不回抽了。这里是一个辅助功能。简单

了解可以了。

if((echange<-MIN_RETRACT && !retracted) ||

(echange>MIN_RETRACT && retracted)) { //move appears to be an attempt to

retract or recover

current_position[E_AXIS] = destination[E_AXIS]; //hide the

slicer-generated retract/recover from calculations

plan_set_e_position(current_position[E_AXIS]); //AND from the

planner

retract(!retracted);

return;

}

}

#endif //FWRETRACT

prepare_move(); //执行移动命令

return;

}

}

从上面的代码来看呢,对于运动类的Gcode,marlin会在process_command()函数

中获取xyze各轴的参数后算出目标坐标(destination[_AXIS]),也会使用

get_coordinates()来获取当前坐标(current_position[E_AXIS])(再次强调,这个坐标是

打印件的世界坐标),当我们知道了目标坐标和当前坐标以后,空间中移动的距离就可以算

出来了(不会算的,请自觉请高中数学老师吃饭去),接下来marlin就使用perpare_move()

来控制电机啦。

接下来呢很自然就要讲讲prepare_move()这个函数啦。先上代码先,代码鸭哥做了精

简,只看关键的部分就是delta和普通结构的代码,先说一下plan_buffer_line()这个函

数的作用的把坐标数组current_position[i] 、 destination[i] 放到一个内存的一个缓存

区里面,然后控制电机转多少圈这样一个作用的,具体代码可以自己去看,在一旦进入这

个函数以后,delta和普通机型的代码都是一样的,也就是说delta和普通结构的电机控制

其实是一样的。

Difference[i]数组 :用来储存目标坐标和当前坐标之间的距离的,(这里是包含了xyze

轴的数组)

Destination[i]数组:目标坐标的数值,是从process_command()函数中G1读取

XYZE参数获取的。

Current_position[i]数组:当前坐标的数值,是从G1 命令中get_coordinates()传递

过来的。如果是3个轴都归零的情况下,current_position就是储存三个坐标原点,如果

开始运动了,这里的值就是上一个prepare_move()循环执行后上一次的destination[i]的

值。(这个下面会有看到赋值语句)

Delta[i]数组:delta打印机的xyz三个电机要移动的距离

void prepare_move()

{

#ifdef DELTA // 设置机子是delta机型(rostock)

float difference[NUM_AXIS]; //定义目标距离,用于转换坐标用的过渡变量

for (int8_t i=0; i < NUM_AXIS; i++) {

difference[i] = destination[i] - current_position[i];

} //计算世界坐标的距离值

//***开始计算笛卡尔距离 并且暴力直线插值来减少运算量***//

float cartesian_mm = sqrt(sq(difference[X_AXIS]) +

sq(difference[Y_AXIS]) +

sq(difference[Z_AXIS]));

if (cartesian_mm < 0.000001) { cartesian_mm = abs(difference[E_AXIS]); }

if (cartesian_mm < 0.000001) { return; }

float seconds = 6000 * cartesian_mm / feedrate / feedmultiply;

int steps = max(1, int(delta_segments_per_second * seconds));

for (int s = 1; s <= steps; s++) {

float fraction = float(s) / float(steps);//直线插值

for(int8_t i=0; i < NUM_AXIS; i++) {

destination[i] = current_position[i] + difference[i] * fraction;

}

//***结束计算笛卡尔距离 并且暴力直线插值来减少运算量***//

calculate_delta(destination);//将打印件的世界坐标转换为xyz电机轴的运动量

plan_buffer_line(delta[X_AXIS], delta[Y_AXIS], delta[Z_AXIS],

destination[E_AXIS], feedrate*feedmultiply/60/100.0,

active_extruder);

}

#endif // DELTA

。。。。。。。。。。。。

#if ! (defined DELTA || defined SCARA)

// Do not use feedmultiply for E or Z only moves

if( (current_position[X_AXIS] == destination [X_AXIS]) &&

(current_position[Y_AXIS] == destination [Y_AXIS])) {

plan_buffer_line(destination[X_AXIS], destination[Y_AXIS],

destination[Z_AXIS], destination[E_AXIS], feedrate/60, active_extruder); //直接将

destination[i]的值发送去运动缓存里面

}

else {

plan_buffer_line(destination[X_AXIS], destination[Y_AXIS], destination[Z_AXIS],

destination[E_AXIS], feedrate*feedmultiply/60/100.0, active_extruder);

}

#endif // !(DELTA || SCARA)

for(int8_t i=0; i < NUM_AXIS; i++) {

current_position[i] = destination[i]; //更新当前坐标的值为刚执行的目标坐标值

}

}

好,看了一大段代码后小结一下。对于普通结构来说,G1 每次将新读取gcode代码

参数传递给prepare_move()函数中destination[i]数组以后,prepare_move()就会将其

传递到plan_buffer_line()进行电机的运动。而delta结构呢,就相对复杂一点,G1命

令读取了gcode代码参数后也是传递到prepare_move()函数中destination[i],然后

marlin要计算目标坐标与当前坐标的笛卡尔距离,然后通过固定时间间隔的方式来将笛卡

尔距离分成若干个小直线,通过这样的方式来就减少cpu的浮点预算量,然后再通过

calculate_delta[i]函数来将简化后的destination[i]换算成三个电机的运动坐标,并传递到

delta[i]中,接下来就是plan_buffer_line()了。

最后!到了最后了!来看看calculate_delta()函数,这个函数的主要用途是将打印

件的世界坐标转换为三个垂直的电机轴的运动坐标哦。注意:新的marlin支持SCARA结

构的delta,那里也有个calculate_delta()的函数,不过那个跟rostock有点差异。所

以我们还是看rostock的吧。

void calculate_delta(float cartesian[3])

{

delta[X_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower1_x-cartesian[X_AXIS])

- sq(delta_tower1_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

delta[Y_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower2_x-cartesian[X_AXIS])

- sq(delta_tower2_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

delta[Z_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower3_x-cartesian[X_AXIS])

- sq(delta_tower3_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

/*

SERIAL_ECHOPGM("cartesian x="); SERIAL_ECHO(cartesian[X_AXIS]);

SERIAL_ECHOPGM(" y="); SERIAL_ECHO(cartesian[Y_AXIS]);

SERIAL_ECHOPGM(" z="); SERIAL_ECHOLN(cartesian[Z_AXIS]);

SERIAL_ECHOPGM("delta x="); SERIAL_ECHO(delta[X_AXIS]);

SERIAL_ECHOPGM(" y="); SERIAL_ECHO(delta[Y_AXIS]);

SERIAL_ECHOPGM(" z="); SERIAL_ECHOLN(delta[Z_AXIS]);

*/

}

代码很简单delta[i]是指电机轴的运动坐标,cartesian[i]是指打印件的世界坐标,从

上面的程序来看就是从prepare_move()中经过插值简化的destination[i]。大伙随便看

一个轴的换算

delta[X_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower1_x-cartesian[X_AXIS])

- sq(delta_tower1_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

delta_diagonal_rod_2 是推杆长的平方

delta_tower1_x 是左前柱的x坐标值,是由radius这个参数算出来的

delta_tower1_y 是左前柱的y坐标值,是由radius这个参数算出来的

具体怎么算就看下面这个函数

void recalc_delta_settings(float radius, float diagonal_rod)

{

delta_tower1_x= -SIN_60*radius; // front left tower

delta_tower1_y= -COS_60*radius;

delta_tower2_x= SIN_60*radius; // front right tower

delta_tower2_y= -COS_60*radius;

delta_tower3_x= 0.0; // back middle tower

delta_tower3_y= radius;

delta_diagonal_rod_2= sq(diagonal_rod);

}

好了回顾一下marlin的delta机型参数是需要什么?

推杆的长度、电机轴上滑块的宽度、喷头支架的宽度,还有三个电机的圆半径。对不

对?忘了?!不要紧,给你看看代码

//================================================

=================

//========================Delta

=============================

Settings

//================================================

=================

// Enable DELTA kinematics and most of the default configuration for Deltas

#define DELTA

// Make delta curves from many straight lines (linear interpolation).

// This is a trade-off between visible corners (not enough segments)

// and processor overload (too many expensive sqrt calls).

#define DELTA_SEGMENTS_PER_SECOND 200

// NOTE NB all values for DELTA_* values MUST be floating point, so always

have a decimal point in them

// Center-to-center distance of the holes in the diagonal push rods.

#define DELTA_DIAGONAL_ROD 250.0 // mm //杆长

// Horizontal offset from middle of printer to smooth rod center.

#define DELTA_SMOOTH_ROD_OFFSET 175.0 // mm //电机轴的圆半径

// Horizontal offset of the universal joints on the end effector.

#define DELTA_EFFECTOR_OFFSET 33.0 // mm // 装喷嘴的平台的中心到杆连接

处的距离

// Horizontal offset of the universal joints on the carriages.

#define DELTA_CARRIAGE_OFFSET 18.0 // mm //电机轴滑块的距离

// Effective horizontal distance bridged by diagonal push rods.

#define DELTA_RADIUS

(DELTA_SMOOTH_ROD_OFFSET-DELTA_EFFECTOR_OFFSET-DELTA_CARRIAGE_OFF

SET)

通过上述的参数可以算出一个DELTA_RADIUS ,这个delta_radius就是上面

“delta_tower1_x 是左前柱的x坐标值,是由radius这个参数算出来的 ”里面的radius

了。

至此所有有关与delta的运动的代码已经通读了一遍。下面就开始分析分析代码和运

动的关系了。

四、

Rostock运动分析

下面是Rostock的结构示意图,分析的第一步是简化整个结构,这里就需要将XY电

机的两个竖轴投影到Z轴的平面上,下图中红色线框画出来的就是z轴的平面,同时我们

可以不考虑XY电机的推杆的运动情况,因为可以XY电机轴的运动可以通过投影在z轴平

面上的虚拟轴笛卡尔空间变换转换回去的。

投影好了以后接下来把z轴放平,那么单独考虑z轴情况,这个情况是在坐标原点的

z电机轴与推杆的情况。为了简化过程,鸭哥已经把z轴滑块,喷头平台都设定为0 了。

那么,z电机轴方向便形成了一个三角形,推杆、radius和z轴电机上的电机坐标,这个

时候三角函数出来啦! 推杆

2

= radius

2

+ 电机坐标

2

在这三角形中推杆是不变的,另外

三角形始终都会是一个直角三角形。一定要记住这几个条件哦。

好,现在我们假设喷头只在x轴上运动,z轴y轴都不动。如果打印件的世界坐标移

动dX距离,rostock需要考虑的问题就是怎么讲dX转换为z轴电机的移动距离了。下图

就是用来表示这种情况。由于推杆长度是不变的,那么

推杆

2

= radius

2

+ 电机坐标

2

就变成下面这样

推杆

2

= (radius +dX)

2

+ (电机坐标-dx)

2

Radius的距离换成destination[x],电机坐标换成delta[x]

推杆

2

= (destination[x +dX])

2

+ (delta[x-dx])

2

Marlin中calculate_delta()这个函数其实就是算

推杆

2

= (destination[x +dX])

2

+ (delta[x-dx])

2

这个等式明白啦,打印件X轴和Y轴的运动分析就明白啦。

另外,再看看打印件z轴的运动分析,还是看看源代码

delta[X_AXIS] = sqrt(delta_diagonal_rod_2

- sq(delta_tower1_x-cartesian[X_AXIS])

- sq(delta_tower1_y-cartesian[Y_AXIS])

) + cartesian[Z_AXIS];

Cartesian[z]是没有在sqrt函数里面的,而是直接加在delta[x]的值上面的。

所以,在调机的时候应该先调z轴,这里就是原因。因为在XYZ三个轴的坐标中只有

Z轴是直接通过同步轮和电机脉冲就可以调准的。调准了z轴以后再调XY轴才是对的。

五、

调机心得

下面是鸭哥的调机的一些心得,与大家分享一下。首先是调机的顺序:

1)选择同步轮可以选择GT2 20齿/40齿 。因为GT2是2mm齿距,整数齿可以使

脉冲数为整数,同时也减少三角函数运算中浮点运算的压力,同时也简化自己的调整步骤。

2)调机时先调z轴运动方向的精度,这个是直接用同步轮和脉冲数就可以调好的。

3)z轴调好以后就调整x轴或者Y轴的,看看运动的路径是不是呈现平面状态,如果

喷头运动路径是弧线这个时候就要调整radius了,增加或者减少radius的值来调整运动

路径是一个平面

4)完成上述步骤以后就可以试打了,这个时候就可以看看XY轴的打印误差是多少了。

如果杆长等硬件参数都比较准确的话那打印误差不会有多少的。如果XY轴有误差的话就

要根据误差大小来等量调整radius这个变量。对应代码是

// Horizontal offset from middle of printer to smooth rod center.

#define DELTA_SMOOTH_ROD_OFFSET 175.0 // mm //电机轴的圆半径

等量修改,比如X轴偏大0.1mm,那么 调整量就是175.0-0.1 = 174.9 了哦。这样

反复几次就基本调好了。

六、

后记

本文是鸭哥的一些心得体会,是个人对marlin还有rostock的一些了解,当然也许会

有这样那样的错误和不足,鸭哥非常欢迎大神高手批评指正。本文首发于珠海创客空间网

站,欢迎转载,著名出处。欢迎修改,修改的地方用另外的文字著名就可以了。

发布评论

评论列表 (0)

  1. 暂无评论