mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-22 00:01:09 -06:00 
			
		
		
		
	Huge speed boost by reducing the number of method calls in GCode.pm
This commit is contained in:
		
							parent
							
								
									36e5e1f933
								
							
						
					
					
						commit
						0049b02bed
					
				
					 5 changed files with 83 additions and 62 deletions
				
			
		|  | @ -117,6 +117,11 @@ sub first_point { | ||||||
|     return $self->polyline->[0]; |     return $self->polyline->[0]; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | sub last_point { | ||||||
|  |     my $self = shift; | ||||||
|  |     return $self->polyline->[-1]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| sub is_perimeter { | sub is_perimeter { | ||||||
|     my $self = shift; |     my $self = shift; | ||||||
|     return $self->role == EXTR_ROLE_PERIMETER |     return $self->role == EXTR_ROLE_PERIMETER | ||||||
|  |  | ||||||
|  | @ -36,7 +36,6 @@ has 'last_speed'         => (is => 'rw', default => sub {""}); | ||||||
| has 'last_f'             => (is => 'rw', default => sub {""}); | has 'last_f'             => (is => 'rw', default => sub {""}); | ||||||
| has 'last_fan_speed'     => (is => 'rw', default => sub {0}); | has 'last_fan_speed'     => (is => 'rw', default => sub {0}); | ||||||
| has 'wipe_path'          => (is => 'rw'); | has 'wipe_path'          => (is => 'rw'); | ||||||
| has 'dec'                => (is => 'ro', default => sub { 3 } ); |  | ||||||
| 
 | 
 | ||||||
| sub _build_speeds { | sub _build_speeds { | ||||||
|     my $self = shift; |     my $self = shift; | ||||||
|  | @ -319,29 +318,89 @@ sub extrude_path { | ||||||
|      |      | ||||||
|     # calculate extrusion length per distance unit |     # calculate extrusion length per distance unit | ||||||
|     my $e = $self->extruder->e_per_mm3 * $area; |     my $e = $self->extruder->e_per_mm3 * $area; | ||||||
|  |     $e = 0 if !$self->config->extrusion_axis; | ||||||
|      |      | ||||||
|     # set speed |     # set speed | ||||||
|     $self->speed( $params{speed} || $role_speeds{$path->role} || die "Unknown role: " . $path->role ); |     $self->speed( $params{speed} || $role_speeds{$path->role} || die "Unknown role: " . $path->role ); | ||||||
|  |     my $F = $self->speeds->{$self->speed} // $self->speed; | ||||||
|  |     if ($self->layer->id == 0) { | ||||||
|  |         $F = $self->config->first_layer_speed =~ /^(\d+(?:\.\d+)?)%$/ | ||||||
|  |             ? sprintf("%.3f", $F * $1/100) | ||||||
|  |             : $self->config->first_layer_speed * 60; | ||||||
|  |     } | ||||||
|      |      | ||||||
|     # extrude arc or line |     # extrude arc or line | ||||||
|  |     $gcode .= ";_BRIDGE_FAN_START\n" if $path->is_bridge; | ||||||
|     my $path_length = 0; |     my $path_length = 0; | ||||||
|     if ($path->isa('Slic3r::ExtrusionPath::Arc')) { |     if ($path->isa('Slic3r::ExtrusionPath::Arc')) { | ||||||
|         $path_length = unscale $path->length; |         $path_length = unscale $path->length; | ||||||
|         $gcode .= $self->G2_G3($path->points->[-1], $path->orientation,  |          | ||||||
|             $path->center, $e * unscale $path_length, $description); |         # calculate extrusion length for this arc | ||||||
|  |         my $E = 0; | ||||||
|  |         if ($e) { | ||||||
|  |             $E = $e * $path_length; | ||||||
|  |             $self->extruder->e(0) if $self->config->use_relative_e_distances; | ||||||
|  |             $self->total_extrusion_length($self->total_extrusion_length + $E); | ||||||
|  |             $E = $self->extruder->e($self->extruder->e + $E); | ||||||
|  |         } | ||||||
|  |          | ||||||
|  |         # compose G-code line | ||||||
|  |         my $point = $path->points->[-1]; | ||||||
|  |         $gcode .= $path->orientation eq 'cw' ? "G2" : "G3"; | ||||||
|  |         $gcode .= sprintf " X%.3f Y%.3f",  | ||||||
|  |             ($point->x * &Slic3r::SCALING_FACTOR) + $self->shift_x - $self->extruder->extruder_offset->[X],  | ||||||
|  |             ($point->y * &Slic3r::SCALING_FACTOR) + $self->shift_y - $self->extruder->extruder_offset->[Y]; #** | ||||||
|  |          | ||||||
|  |         # XY distance of the center from the start position | ||||||
|  |         $gcode .= sprintf " I%.3f J%.3f", | ||||||
|  |             ($path->center->[X] - $self->last_pos->[X]) * &Slic3r::SCALING_FACTOR, | ||||||
|  |             ($path->center->[Y] - $self->last_pos->[Y]) * &Slic3r::SCALING_FACTOR; | ||||||
|  |          | ||||||
|  |         $gcode .= sprintf(" %s%.5f", $self->config->extrusion_axis, $E) | ||||||
|  |             if $E; | ||||||
|  |         $gcode .= " F$F"; | ||||||
|  |         $gcode .= " ; $description" | ||||||
|  |             if $self->config->gcode_comments; | ||||||
|  |         $gcode .= "\n"; | ||||||
|  |          | ||||||
|         $self->wipe_path(undef); |         $self->wipe_path(undef); | ||||||
|     } else { |     } else { | ||||||
|  |         my $local_F = $F; | ||||||
|         foreach my $line ($path->lines) { |         foreach my $line ($path->lines) { | ||||||
|             my $line_length = unscale $line->length; |             $path_length += my $line_length = unscale $line->length; | ||||||
|             $path_length += $line_length; |              | ||||||
|             $gcode .= $self->G1($line->[B], undef, $e * $line_length, $description); |             # calculate extrusion length for this line | ||||||
|  |             my $E = 0; | ||||||
|  |             if ($e) { | ||||||
|  |                 $E = $e * $line_length; | ||||||
|  |                 $self->extruder->e(0) if $self->config->use_relative_e_distances; | ||||||
|  |                 $self->total_extrusion_length($self->total_extrusion_length + $E); | ||||||
|  |                 $E = $self->extruder->e($self->extruder->e + $E); | ||||||
|  |             } | ||||||
|  |              | ||||||
|  |             # compose G-code line | ||||||
|  |             $gcode .= sprintf "G1 X%.3f Y%.3f", | ||||||
|  |                 ($line->[B]->x * &Slic3r::SCALING_FACTOR) + $self->shift_x - $self->extruder->extruder_offset->[X],  | ||||||
|  |                 ($line->[B]->y * &Slic3r::SCALING_FACTOR) + $self->shift_y - $self->extruder->extruder_offset->[Y];  #** | ||||||
|  |             $gcode .= sprintf(" %s%.5f", $self->config->extrusion_axis, $E) | ||||||
|  |                 if $E; | ||||||
|  |             $gcode .= " F$local_F" | ||||||
|  |                 if $local_F; | ||||||
|  |             $gcode .= " ; $description" | ||||||
|  |                 if $self->config->gcode_comments; | ||||||
|  |             $gcode .= "\n"; | ||||||
|  |              | ||||||
|  |             # only include F in the first line | ||||||
|  |             $local_F = 0; | ||||||
|         } |         } | ||||||
|         $self->wipe_path(Slic3r::Polyline->new(reverse @{$path->points})) |         $self->wipe_path(Slic3r::Polyline->new(reverse @{$path->points})) | ||||||
|             if $self->enable_wipe; |             if $self->enable_wipe; | ||||||
|     } |     } | ||||||
|  |     $gcode .= ";_BRIDGE_FAN_END\n" if $path->is_bridge; | ||||||
|  |     $self->last_pos($path->last_point); | ||||||
|      |      | ||||||
|     if ($self->config->cooling) { |     if ($self->config->cooling) { | ||||||
|         my $path_time = $path_length / $self->speeds->{$self->last_speed} * 60; |         my $path_time = $path_length / $F * 60; | ||||||
|         if ($self->layer->id == 0) { |         if ($self->layer->id == 0) { | ||||||
|             $path_time = $self->config->first_layer_speed =~ /^(\d+(?:\.\d+)?)%$/ |             $path_time = $self->config->first_layer_speed =~ /^(\d+(?:\.\d+)?)%$/ | ||||||
|                 ? $path_time / ($1/100) |                 ? $path_time / ($1/100) | ||||||
|  | @ -577,69 +636,30 @@ sub G1 { | ||||||
| 
 | 
 | ||||||
| sub _G0_G1 { | sub _G0_G1 { | ||||||
|     my ($self, $gcode, $point, $z, $e, $comment) = @_; |     my ($self, $gcode, $point, $z, $e, $comment) = @_; | ||||||
|     my $dec = $self->dec; |  | ||||||
|      |      | ||||||
|     if ($point) { |     if ($point) { | ||||||
|         $gcode .= sprintf " X%.${dec}f Y%.${dec}f",  |         $gcode .= sprintf " X%.3f Y%.3f",  | ||||||
|             ($point->x * &Slic3r::SCALING_FACTOR) + $self->shift_x - $self->extruder->extruder_offset->[X],  |             ($point->x * &Slic3r::SCALING_FACTOR) + $self->shift_x - $self->extruder->extruder_offset->[X],  | ||||||
|             ($point->y * &Slic3r::SCALING_FACTOR) + $self->shift_y - $self->extruder->extruder_offset->[Y]; #** |             ($point->y * &Slic3r::SCALING_FACTOR) + $self->shift_y - $self->extruder->extruder_offset->[Y]; #** | ||||||
|         $self->last_pos($point->clone); |         $self->last_pos($point->clone); | ||||||
|     } |     } | ||||||
|     if (defined $z && (!defined $self->z || $z != $self->z)) { |     if (defined $z && (!defined $self->z || $z != $self->z)) { | ||||||
|         $self->z($z); |         $self->z($z); | ||||||
|         $gcode .= sprintf " Z%.${dec}f", $z; |         $gcode .= sprintf " Z%.3f", $z; | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     return $self->_Gx($gcode, $e, $comment); |     return $self->_Gx($gcode, $e, $comment); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| sub G2_G3 { |  | ||||||
|     my ($self, $point, $orientation, $center, $e, $comment) = @_; |  | ||||||
|     my $dec = $self->dec; |  | ||||||
|      |  | ||||||
|     my $gcode = $orientation eq 'cw' ? "G2" : "G3"; |  | ||||||
|      |  | ||||||
|     $gcode .= sprintf " X%.${dec}f Y%.${dec}f",  |  | ||||||
|         ($point->x * &Slic3r::SCALING_FACTOR) + $self->shift_x - $self->extruder->extruder_offset->[X],  |  | ||||||
|         ($point->y * &Slic3r::SCALING_FACTOR) + $self->shift_y - $self->extruder->extruder_offset->[Y]; #** |  | ||||||
|      |  | ||||||
|     # XY distance of the center from the start position |  | ||||||
|     $gcode .= sprintf " I%.${dec}f J%.${dec}f", |  | ||||||
|         ($center->[X] - $self->last_pos->[X]) * &Slic3r::SCALING_FACTOR, |  | ||||||
|         ($center->[Y] - $self->last_pos->[Y]) * &Slic3r::SCALING_FACTOR; |  | ||||||
|      |  | ||||||
|     $self->last_pos($point); |  | ||||||
|     return $self->_Gx($gcode, $e, $comment); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| sub _Gx { | sub _Gx { | ||||||
|     my ($self, $gcode, $e, $comment) = @_; |     my ($self, $gcode, $e, $comment) = @_; | ||||||
|     my $dec = $self->dec; |  | ||||||
|      |      | ||||||
|     # output speed if it's different from last one used |     my $F = $self->speed eq 'retract' | ||||||
|     # (goal: reduce gcode size) |         ? ($self->extruder->retract_speed_mm_min) | ||||||
|     my $append_bridge_off = 0; |         : $self->speeds->{$self->speed} // $self->speed; | ||||||
|     my $F; |     $self->last_speed($self->speed); | ||||||
|     if ($self->speed ne $self->last_speed) { |     $self->last_f($F); | ||||||
|         if ($self->speed eq 'bridge') { |     $gcode .= sprintf " F%.3f", $F; | ||||||
|             $gcode = ";_BRIDGE_FAN_START\n$gcode"; |  | ||||||
|         } elsif ($self->last_speed eq 'bridge') { |  | ||||||
|             $append_bridge_off = 1; |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         # apply the speed reduction for print moves on bottom layer |  | ||||||
|         $F = $self->speed eq 'retract' |  | ||||||
|             ? ($self->extruder->retract_speed_mm_min) |  | ||||||
|             : $self->speeds->{$self->speed} // $self->speed; |  | ||||||
|         if ($e && $self->layer && $self->layer->id == 0 && $comment !~ /retract/) { |  | ||||||
|             $F = $self->config->first_layer_speed =~ /^(\d+(?:\.\d+)?)%$/ |  | ||||||
|                 ? ($F * $1/100) |  | ||||||
|                 : $self->config->first_layer_speed * 60; |  | ||||||
|         } |  | ||||||
|         $self->last_speed($self->speed); |  | ||||||
|         $self->last_f($F); |  | ||||||
|     } |  | ||||||
|     $gcode .= sprintf " F%.${dec}f", $F if defined $F; |  | ||||||
|      |      | ||||||
|     # output extrusion distance |     # output extrusion distance | ||||||
|     if ($e && $self->config->extrusion_axis) { |     if ($e && $self->config->extrusion_axis) { | ||||||
|  | @ -649,10 +669,7 @@ sub _Gx { | ||||||
|         $gcode .= sprintf " %s%.5f", $self->config->extrusion_axis, $self->extruder->e; |         $gcode .= sprintf " %s%.5f", $self->config->extrusion_axis, $self->extruder->e; | ||||||
|     } |     } | ||||||
|      |      | ||||||
|     $gcode .= sprintf " ; %s", $comment if $comment && $self->config->gcode_comments; |     $gcode .= " ; $comment" if $comment && $self->config->gcode_comments; | ||||||
|     if ($append_bridge_off) { |  | ||||||
|         $gcode = ";_BRIDGE_FAN_END\n$gcode"; |  | ||||||
|     } |  | ||||||
|     return "$gcode\n"; |     return "$gcode\n"; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -56,10 +56,9 @@ sub flush { | ||||||
|         Slic3r::debugf "  fan = %d%%, speed = %d%%\n", $fan_speed, $speed_factor * 100; |         Slic3r::debugf "  fan = %d%%, speed = %d%%\n", $fan_speed, $speed_factor * 100; | ||||||
|          |          | ||||||
|         if ($speed_factor < 1) { |         if ($speed_factor < 1) { | ||||||
|             my $dec = $self->gcodegen->dec; |  | ||||||
|             $gcode =~ s/^(?=.*? [XY])(?=.*? E)(?!;_WIPE)(?<!;_BRIDGE_FAN_START\n)(G1 .*?F)(\d+(?:\.\d+)?)/ |             $gcode =~ s/^(?=.*? [XY])(?=.*? E)(?!;_WIPE)(?<!;_BRIDGE_FAN_START\n)(G1 .*?F)(\d+(?:\.\d+)?)/ | ||||||
|                 my $new_speed = $2 * $speed_factor; |                 my $new_speed = $2 * $speed_factor; | ||||||
|                 $1 . sprintf("%.${dec}f", $new_speed < $self->min_print_speed ? $self->min_print_speed : $new_speed) |                 $1 . sprintf("%.3f", $new_speed < $self->min_print_speed ? $self->min_print_speed : $new_speed) | ||||||
|                 /gexm; |                 /gexm; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -49,7 +49,7 @@ sub parse { | ||||||
|         } |         } | ||||||
|          |          | ||||||
|         # run callback |         # run callback | ||||||
|         #$cb->($self, $command, \%args, \%info); |         $cb->($self, $command, \%args, \%info); | ||||||
|          |          | ||||||
|         # update coordinates |         # update coordinates | ||||||
|         if ($command =~ /^(?:G[01]|G92)$/) { |         if ($command =~ /^(?:G[01]|G92)$/) { | ||||||
|  |  | ||||||
|  | @ -58,7 +58,7 @@ my $test = sub { | ||||||
|         my $one_axis_would_trigger_limit_without_pause = 0; |         my $one_axis_would_trigger_limit_without_pause = 0; | ||||||
|         foreach my $axis (qw(X Y)) { |         foreach my $axis (qw(X Y)) { | ||||||
|             # are we changing direction on this axis? |             # are we changing direction on this axis? | ||||||
|             my $dir = $info->{"dist_$axis"} <=> 0; |             my $dir = $info->{"dist_$axis"} <=> ($args->{$axis} // $self->$axis); | ||||||
|             if ($dir != 0 && $dir{$axis} != $dir) { |             if ($dir != 0 && $dir{$axis} != $dir) { | ||||||
|                 # this move changes direction on this axis |                 # this move changes direction on this axis | ||||||
|                 if ($dir{$axis} != 0) { |                 if ($dir{$axis} != 0) { | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Alessandro Ranellucci
						Alessandro Ranellucci