2011-09-01 19:06:28 +00:00
|
|
|
package Slic3r;
|
|
|
|
|
2012-01-18 19:04:18 +00:00
|
|
|
# Copyright holder: Alessandro Ranellucci
|
|
|
|
# This application is licensed under the GNU Affero General Public License, version 3
|
|
|
|
|
2011-09-01 19:06:28 +00:00
|
|
|
use strict;
|
|
|
|
use warnings;
|
2012-01-22 12:56:15 +00:00
|
|
|
require v5.10;
|
2011-09-01 19:06:28 +00:00
|
|
|
|
2014-08-04 10:43:18 +00:00
|
|
|
our $VERSION = "1.2.0";
|
2011-11-13 21:57:58 +00:00
|
|
|
|
2011-09-05 11:33:09 +00:00
|
|
|
our $debug = 0;
|
|
|
|
sub debugf {
|
|
|
|
printf @_ if $debug;
|
|
|
|
}
|
|
|
|
|
2012-05-19 18:25:59 +00:00
|
|
|
# load threads before Moo as required by it
|
2012-05-21 17:16:42 +00:00
|
|
|
our $have_threads;
|
2012-05-21 16:44:31 +00:00
|
|
|
BEGIN {
|
|
|
|
use Config;
|
2013-02-27 10:08:08 +00:00
|
|
|
$have_threads = $Config{useithreads} && eval "use threads; use threads::shared; use Thread::Queue; 1";
|
2013-07-24 08:06:02 +00:00
|
|
|
|
|
|
|
### temporarily disable threads if using the broken Moo version
|
|
|
|
use Moo;
|
|
|
|
$have_threads = 0 if $Moo::VERSION == 1.003000;
|
2012-05-21 16:44:31 +00:00
|
|
|
}
|
2012-05-19 18:25:59 +00:00
|
|
|
|
2012-09-11 14:02:26 +00:00
|
|
|
warn "Running Slic3r under Perl >= 5.16 is not supported nor recommended\n"
|
|
|
|
if $^V >= v5.16;
|
|
|
|
|
2012-05-29 12:19:14 +00:00
|
|
|
use FindBin;
|
|
|
|
our $var = "$FindBin::Bin/var";
|
|
|
|
|
2013-01-13 09:18:34 +00:00
|
|
|
use Encode;
|
|
|
|
use Encode::Locale;
|
2013-09-12 09:09:03 +00:00
|
|
|
use Moo 1.003001;
|
2013-01-13 09:18:34 +00:00
|
|
|
|
2013-07-14 11:05:55 +00:00
|
|
|
use Slic3r::XS; # import all symbols (constants etc.) before they get parsed
|
2011-10-03 09:55:32 +00:00
|
|
|
use Slic3r::Config;
|
2011-10-15 09:36:05 +00:00
|
|
|
use Slic3r::ExPolygon;
|
2011-09-26 09:42:08 +00:00
|
|
|
use Slic3r::Extruder;
|
2011-09-25 21:15:45 +00:00
|
|
|
use Slic3r::ExtrusionLoop;
|
2011-09-04 10:04:01 +00:00
|
|
|
use Slic3r::ExtrusionPath;
|
2011-09-26 08:52:58 +00:00
|
|
|
use Slic3r::ExtrusionPath::Collection;
|
2011-09-05 10:21:27 +00:00
|
|
|
use Slic3r::Fill;
|
2012-06-06 16:05:03 +00:00
|
|
|
use Slic3r::Flow;
|
2012-02-25 16:35:25 +00:00
|
|
|
use Slic3r::Format::AMF;
|
2012-05-20 09:40:37 +00:00
|
|
|
use Slic3r::Format::OBJ;
|
2012-02-25 16:35:25 +00:00
|
|
|
use Slic3r::Format::STL;
|
2012-06-28 12:44:54 +00:00
|
|
|
use Slic3r::GCode;
|
2013-08-28 17:50:16 +00:00
|
|
|
use Slic3r::GCode::ArcFitting;
|
2013-04-03 23:17:44 +00:00
|
|
|
use Slic3r::GCode::CoolingBuffer;
|
2013-05-18 14:57:44 +00:00
|
|
|
use Slic3r::GCode::Layer;
|
2012-08-23 13:42:58 +00:00
|
|
|
use Slic3r::GCode::MotionPlanner;
|
2014-03-16 23:39:07 +00:00
|
|
|
use Slic3r::GCode::PlaceholderParser;
|
2013-05-13 18:14:33 +00:00
|
|
|
use Slic3r::GCode::Reader;
|
2013-05-13 18:15:45 +00:00
|
|
|
use Slic3r::GCode::SpiralVase;
|
2013-08-28 14:51:58 +00:00
|
|
|
use Slic3r::GCode::VibrationLimit;
|
2011-12-07 18:33:59 +00:00
|
|
|
use Slic3r::Geometry qw(PI);
|
2013-06-07 21:16:02 +00:00
|
|
|
use Slic3r::Geometry::Clipper;
|
2011-09-01 19:06:28 +00:00
|
|
|
use Slic3r::Layer;
|
2014-04-07 21:18:11 +00:00
|
|
|
use Slic3r::Layer::BridgeDetector;
|
2012-09-23 00:52:31 +00:00
|
|
|
use Slic3r::Layer::Region;
|
2011-09-01 19:06:28 +00:00
|
|
|
use Slic3r::Line;
|
2012-08-29 14:49:38 +00:00
|
|
|
use Slic3r::Model;
|
2011-09-01 19:06:28 +00:00
|
|
|
use Slic3r::Point;
|
2011-10-15 09:36:05 +00:00
|
|
|
use Slic3r::Polygon;
|
2011-09-01 19:06:28 +00:00
|
|
|
use Slic3r::Polyline;
|
|
|
|
use Slic3r::Print;
|
2012-04-29 10:51:20 +00:00
|
|
|
use Slic3r::Print::Object;
|
2014-01-02 16:54:18 +00:00
|
|
|
use Slic3r::Print::Simple;
|
2013-09-17 21:27:57 +00:00
|
|
|
use Slic3r::Print::SupportMaterial;
|
2011-09-01 19:06:28 +00:00
|
|
|
use Slic3r::Surface;
|
2011-11-27 10:40:03 +00:00
|
|
|
use Slic3r::TriangleMesh;
|
2013-04-27 18:55:43 +00:00
|
|
|
our $build = eval "use Slic3r::Build; 1";
|
2014-07-12 09:41:18 +00:00
|
|
|
use Thread::Semaphore;
|
2011-09-01 19:06:28 +00:00
|
|
|
|
2012-07-27 19:13:03 +00:00
|
|
|
use constant SCALING_FACTOR => 0.000001;
|
2012-11-18 16:38:08 +00:00
|
|
|
use constant RESOLUTION => 0.0125;
|
|
|
|
use constant SCALED_RESOLUTION => RESOLUTION / SCALING_FACTOR;
|
2012-07-27 19:13:03 +00:00
|
|
|
use constant SMALL_PERIMETER_LENGTH => (6.5 / SCALING_FACTOR) * 2 * PI;
|
2014-01-03 19:02:00 +00:00
|
|
|
use constant LOOP_CLIPPING_LENGTH_OVER_NOZZLE_DIAMETER => 0.15;
|
2013-03-16 23:57:58 +00:00
|
|
|
use constant INFILL_OVERLAP_OVER_SPACING => 0.45;
|
2013-08-13 07:45:33 +00:00
|
|
|
use constant EXTERNAL_INFILL_MARGIN => 3;
|
2014-04-14 23:41:40 +00:00
|
|
|
use constant INSET_OVERLAP_TOLERANCE => 0.2;
|
2012-07-27 19:13:03 +00:00
|
|
|
|
2014-06-13 17:23:51 +00:00
|
|
|
# keep track of threads we created
|
2014-07-01 17:00:23 +00:00
|
|
|
my @threads : shared = ();
|
2014-07-12 09:41:18 +00:00
|
|
|
my $sema = Thread::Semaphore->new;
|
|
|
|
my $paused = 0;
|
2014-07-01 17:00:23 +00:00
|
|
|
|
|
|
|
sub spawn_thread {
|
|
|
|
my ($cb) = @_;
|
|
|
|
|
2014-07-12 09:41:18 +00:00
|
|
|
@_ = ();
|
|
|
|
my $thread = threads->create(sub {
|
|
|
|
local $SIG{'KILL'} = sub {
|
|
|
|
Slic3r::debugf "Exiting thread...\n";
|
|
|
|
Slic3r::thread_cleanup();
|
|
|
|
threads->exit();
|
|
|
|
};
|
|
|
|
local $SIG{'STOP'} = sub {
|
|
|
|
$sema->down;
|
|
|
|
$sema->up;
|
|
|
|
};
|
|
|
|
$cb->();
|
|
|
|
});
|
2014-07-01 17:00:23 +00:00
|
|
|
push @threads, $thread->tid;
|
|
|
|
return $thread;
|
|
|
|
}
|
2014-06-13 17:23:51 +00:00
|
|
|
|
2012-04-09 10:29:47 +00:00
|
|
|
sub parallelize {
|
|
|
|
my %params = @_;
|
|
|
|
|
2013-12-24 10:29:31 +00:00
|
|
|
if (!$params{disable} && $Slic3r::have_threads && $params{threads} > 1) {
|
2012-09-22 17:04:36 +00:00
|
|
|
my @items = (ref $params{items} eq 'CODE') ? $params{items}->() : @{$params{items}};
|
2012-04-09 10:29:47 +00:00
|
|
|
my $q = Thread::Queue->new;
|
2013-12-24 10:29:31 +00:00
|
|
|
$q->enqueue(@items, (map undef, 1..$params{threads}));
|
2012-04-09 10:29:47 +00:00
|
|
|
|
2013-07-07 16:43:04 +00:00
|
|
|
my $thread_cb = sub {
|
2014-06-13 17:23:51 +00:00
|
|
|
# execute thread callback
|
2013-11-12 11:09:33 +00:00
|
|
|
$params{thread_cb}->($q);
|
2014-06-13 17:23:51 +00:00
|
|
|
|
|
|
|
# cleanup before terminating thread
|
2013-07-11 14:17:36 +00:00
|
|
|
Slic3r::thread_cleanup();
|
2013-11-12 11:09:33 +00:00
|
|
|
|
|
|
|
# This explicit exit avoids an untrappable
|
|
|
|
# "Attempt to free unreferenced scalar" error
|
|
|
|
# triggered on Ubuntu 12.04 32-bit when we're running
|
|
|
|
# from the Wx plater and
|
|
|
|
# we're reusing the same plater object more than once.
|
|
|
|
# The downside to using this exit is that we can't return
|
|
|
|
# any value to the main thread but we're not doing that
|
|
|
|
# anymore anyway.
|
|
|
|
# collect_cb is completely useless now
|
|
|
|
# and should be removed from the codebase.
|
|
|
|
threads->exit;
|
2013-07-07 16:43:04 +00:00
|
|
|
};
|
2013-09-17 21:14:49 +00:00
|
|
|
$params{collect_cb} ||= sub {};
|
2013-07-07 16:43:04 +00:00
|
|
|
|
2013-05-19 15:34:33 +00:00
|
|
|
@_ = ();
|
2014-07-01 17:00:23 +00:00
|
|
|
my @my_threads = map spawn_thread($thread_cb), 1..$params{threads};
|
2014-06-13 17:23:51 +00:00
|
|
|
foreach my $th (@my_threads) {
|
2012-04-09 10:29:47 +00:00
|
|
|
$params{collect_cb}->($th->join);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$params{no_threads_cb}->();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-11 14:17:36 +00:00
|
|
|
# call this at the very end of each thread (except the main one)
|
|
|
|
# so that it does not try to free existing objects.
|
|
|
|
# at that stage, existing objects are only those that we
|
|
|
|
# inherited at the thread creation (thus shared) and those
|
|
|
|
# that we are returning: destruction will be handled by the
|
|
|
|
# main thread in both cases.
|
2013-09-17 21:14:49 +00:00
|
|
|
# reminder: do not destroy inherited objects in other threads,
|
|
|
|
# as the main thread will still try to destroy them when they
|
|
|
|
# go out of scope; in other words, if you're undef()'ing an
|
|
|
|
# object in a thread, make sure the main thread still holds a
|
|
|
|
# reference so that it won't be destroyed in thread.
|
2013-07-11 14:17:36 +00:00
|
|
|
sub thread_cleanup {
|
2014-06-13 17:23:51 +00:00
|
|
|
return if !$Slic3r::have_threads;
|
|
|
|
|
2013-07-11 14:17:36 +00:00
|
|
|
# prevent destruction of shared objects
|
|
|
|
no warnings 'redefine';
|
2013-12-21 23:39:03 +00:00
|
|
|
*Slic3r::Config::DESTROY = sub {};
|
2014-01-02 09:44:54 +00:00
|
|
|
*Slic3r::Config::Full::DESTROY = sub {};
|
2013-12-21 23:39:03 +00:00
|
|
|
*Slic3r::Config::Print::DESTROY = sub {};
|
2014-01-02 09:44:54 +00:00
|
|
|
*Slic3r::Config::PrintObject::DESTROY = sub {};
|
|
|
|
*Slic3r::Config::PrintRegion::DESTROY = sub {};
|
2013-07-16 07:49:34 +00:00
|
|
|
*Slic3r::ExPolygon::DESTROY = sub {};
|
2013-07-13 22:38:01 +00:00
|
|
|
*Slic3r::ExPolygon::Collection::DESTROY = sub {};
|
2014-04-07 23:42:29 +00:00
|
|
|
*Slic3r::Extruder::DESTROY = sub {};
|
2013-07-16 07:49:34 +00:00
|
|
|
*Slic3r::ExtrusionLoop::DESTROY = sub {};
|
|
|
|
*Slic3r::ExtrusionPath::DESTROY = sub {};
|
2013-07-18 20:29:12 +00:00
|
|
|
*Slic3r::ExtrusionPath::Collection::DESTROY = sub {};
|
2014-01-14 19:11:08 +00:00
|
|
|
*Slic3r::Flow::DESTROY = sub {};
|
2014-05-06 08:07:18 +00:00
|
|
|
*Slic3r::GCode::PlaceholderParser::DESTROY = sub {};
|
2014-01-14 19:11:08 +00:00
|
|
|
*Slic3r::Geometry::BoundingBox::DESTROY = sub {};
|
|
|
|
*Slic3r::Geometry::BoundingBoxf3::DESTROY = sub {};
|
2013-07-16 07:49:34 +00:00
|
|
|
*Slic3r::Line::DESTROY = sub {};
|
2014-04-29 23:04:49 +00:00
|
|
|
*Slic3r::Model::DESTROY = sub {};
|
|
|
|
*Slic3r::Model::Object::DESTROY = sub {};
|
2013-07-16 07:49:34 +00:00
|
|
|
*Slic3r::Point::DESTROY = sub {};
|
2014-04-28 20:02:34 +00:00
|
|
|
*Slic3r::Pointf::DESTROY = sub {};
|
2014-01-14 19:11:08 +00:00
|
|
|
*Slic3r::Pointf3::DESTROY = sub {};
|
2013-07-16 07:49:34 +00:00
|
|
|
*Slic3r::Polygon::DESTROY = sub {};
|
|
|
|
*Slic3r::Polyline::DESTROY = sub {};
|
2013-09-12 09:09:03 +00:00
|
|
|
*Slic3r::Polyline::Collection::DESTROY = sub {};
|
2014-05-06 08:07:18 +00:00
|
|
|
*Slic3r::Print::DESTROY = sub {};
|
|
|
|
*Slic3r::Print::Region::DESTROY = sub {};
|
2013-07-16 07:49:34 +00:00
|
|
|
*Slic3r::Surface::DESTROY = sub {};
|
|
|
|
*Slic3r::Surface::Collection::DESTROY = sub {};
|
2013-09-12 09:09:03 +00:00
|
|
|
*Slic3r::TriangleMesh::DESTROY = sub {};
|
2013-09-19 14:00:47 +00:00
|
|
|
return undef; # this prevents a "Scalars leaked" warning
|
2013-07-11 14:17:36 +00:00
|
|
|
}
|
|
|
|
|
2014-07-12 09:41:18 +00:00
|
|
|
sub get_running_threads {
|
|
|
|
return grep defined($_), map threads->object($_), @threads;
|
|
|
|
}
|
|
|
|
|
2014-07-01 17:00:23 +00:00
|
|
|
sub kill_all_threads {
|
|
|
|
# detach any running thread created in the current one
|
2014-07-04 10:12:15 +00:00
|
|
|
my @killed = ();
|
2014-07-12 09:41:18 +00:00
|
|
|
foreach my $thread (get_running_threads()) {
|
2014-07-01 17:00:23 +00:00
|
|
|
$thread->kill('KILL');
|
2014-07-04 10:12:15 +00:00
|
|
|
push @killed, $thread;
|
2014-07-01 17:00:23 +00:00
|
|
|
}
|
2014-07-12 09:41:18 +00:00
|
|
|
|
|
|
|
# unlock semaphore before we block on wait
|
|
|
|
# otherwise we'd get a deadlock if threads were paused
|
|
|
|
resume_threads();
|
2014-07-04 10:12:15 +00:00
|
|
|
$_->join for @killed; # block until threads are killed
|
2014-07-01 17:00:23 +00:00
|
|
|
@threads = ();
|
|
|
|
}
|
|
|
|
|
2014-07-12 09:41:18 +00:00
|
|
|
sub pause_threads {
|
|
|
|
return if $paused;
|
|
|
|
$paused = 1;
|
|
|
|
$sema->down;
|
|
|
|
$_->kill('STOP') for get_running_threads();
|
|
|
|
}
|
|
|
|
|
|
|
|
sub resume_threads {
|
|
|
|
return unless $paused;
|
|
|
|
$paused = 0;
|
|
|
|
$sema->up;
|
|
|
|
}
|
|
|
|
|
2013-02-22 17:40:00 +00:00
|
|
|
sub encode_path {
|
|
|
|
my ($filename) = @_;
|
|
|
|
return encode('locale_fs', $filename);
|
|
|
|
}
|
|
|
|
|
2013-01-13 09:18:34 +00:00
|
|
|
sub open {
|
|
|
|
my ($fh, $mode, $filename) = @_;
|
2013-02-22 17:40:00 +00:00
|
|
|
return CORE::open $$fh, $mode, encode_path($filename);
|
2013-01-13 09:18:34 +00:00
|
|
|
}
|
|
|
|
|
2013-12-13 11:18:30 +00:00
|
|
|
# this package declaration prevents an ugly fatal warning to be emitted when
|
|
|
|
# spawning a new thread
|
|
|
|
package GLUquadricObjPtr;
|
|
|
|
|
2011-09-01 19:06:28 +00:00
|
|
|
1;
|