diff --git a/.github/workflows/benchmark.yml b/.github/workflows/benchmark.yml index 059ef365cf..ee49634a2a 100644 --- a/.github/workflows/benchmark.yml +++ b/.github/workflows/benchmark.yml @@ -37,10 +37,10 @@ jobs: benchmark: uses: ultimaker/cura-workflows/.github/workflows/benchmark.yml@main with: - conan_extra_args: "-o \"curaengine/*:enable_benchmarks=True\"" - benchmark_cmd: "./build/Release/benchmark/benchmarks --benchmark_format=json --benchmark_out=benchmark_result.json" + conan_extra_args: "-c tools.build:skip_test=False -o \"curaengine/*:enable_benchmarks=True\"" + benchmark_cmd: "benchmark/benchmarks --benchmark_format=json --benchmark_out=benchmark_result.json" name: "C++ Benchmark" - output_file_path: "build/Release/benchmark_result.json" + output_file_path: "benchmark_result.json" data_dir: "dev/bench" tool: "googlecpp" alert_threshold: "150%" diff --git a/.github/workflows/gcodeanalyzer.yml b/.github/workflows/gcodeanalyzer.yml index be45106c2c..8145b07e83 100644 --- a/.github/workflows/gcodeanalyzer.yml +++ b/.github/workflows/gcodeanalyzer.yml @@ -24,20 +24,13 @@ permissions: contents: write deployments: write -# TODO: Make cura-workflows/benchmark.yml generic enough to fit the gcodeanalyzer benchmark jobs: check_actor: uses: ultimaker/cura-workflows/.github/workflows/check-actor.yml@main secrets: inherit - conan-recipe-version: - needs: [ check_actor ] - if: ${{ needs.check_actor.outputs.proceed == 'true' }} - uses: ultimaker/cura-workflows/.github/workflows/conan-recipe-version.yml@main - gcodeanalyzer: - needs: [ conan-recipe-version ] name: Run GCodeAnalyzer on the engine runs-on: ubuntu-latest steps: @@ -64,17 +57,19 @@ jobs: fetch-depth: 1 token: ${{ secrets.GITHUB_TOKEN }} - - name: Install dependencies - run: conan install . ${{ needs.conan-recipe-version.outputs.version_full }} -s "*:build_type=Release" --build=missing --update -g VirtualRunEnv -c tools.build:skip_test=True -o with_cura_resources=True - - - name: Build CuraEngine and tests + - name: Install Python requirements run: | - source build/Release/generators/conanbuild.sh - cmake --preset release - cmake --build --preset release + pip install pandas + pip install git+https://github.com/ultimaker/libcharon@CURA-9495_analyzer_requisites#egg=charon + + - name: Install dependencies and build CuraEngine + run: conan build . -s build_type=Release --build=missing --update -g VirtualRunEnv -o "curaengine/*:with_cura_resources=True" - name: Collect STL-files, run CuraEngine, output GCode-files run: | + source build/Release/generators/conanrun.sh + echo $CURA_RESOURCES + ls $CURA_RESOURCES for file in `ls NightlyTestModels/*.stl`; do ( time ./build/Release/CuraEngine slice --force-read-parent --force-read-nondefault -v -p -j $CURA_RESOURCES/definitions/ultimaker_s3.def.json -l $file -o `basename $file .stl`.gcode ) 2> `basename $file .stl`.time @@ -83,6 +78,8 @@ jobs: # TODO: Move this to GCodeAnalyzer - name: Run GCodeAnalyzer on generated GCode files id: gcode_out + shell: python + working-directory: GCodeAnalyzer run: | import sys import os @@ -205,8 +202,6 @@ jobs: with open("../output.json", "w") as outfile: outfile.write(json.dumps(jzon)) - shell: python - working-directory: GCodeAnalyzer - name: Store benchmark result uses: benchmark-action/github-action-benchmark@v1 diff --git a/.github/workflows/stress_benchmark.yml b/.github/workflows/stress_benchmark.yml index 56432c9899..a76be61f33 100644 --- a/.github/workflows/stress_benchmark.yml +++ b/.github/workflows/stress_benchmark.yml @@ -27,30 +27,19 @@ permissions: contents: write deployments: write -env: - CONAN_LOGIN_USERNAME: ${{ secrets.CONAN_USER }} - CONAN_PASSWORD: ${{ secrets.CONAN_PASS }} - jobs: check_actor: uses: ultimaker/cura-workflows/.github/workflows/check-actor.yml@main secrets: inherit - conan-recipe-version: - needs: [ check_actor ] - if: ${{ needs.check_actor.outputs.proceed == 'true' }} - uses: ultimaker/cura-workflows/.github/workflows/conan-recipe-version.yml@main - benchmark: - needs: [ conan-recipe-version ] uses: ultimaker/cura-workflows/.github/workflows/benchmark.yml@main with: - recipe_id_full: ${{ needs.conan-recipe-version.outputs.recipe_id_full }} - conan_extra_args: "-o curaengine:enable_benchmarks=True" + conan_extra_args: "-c tools.build:skip_test=False -o \"curaengine/*:enable_benchmarks=True\"" benchmark_cmd: "stress_benchmark/stress_benchmark -o benchmark_result.json" name: "Stress Benchmark" - output_file_path: "build/Release/benchmark_result.json" + output_file_path: "benchmark_result.json" data_dir: "dev/stress_bench" tool: "customSmallerIsBetter" secrets: inherit diff --git a/include/PathOrderOptimizer.h b/include/PathOrderOptimizer.h index 2d339e79f8..f50482d4ad 100644 --- a/include/PathOrderOptimizer.h +++ b/include/PathOrderOptimizer.h @@ -738,29 +738,26 @@ class PathOrderOptimizer BestElementFinder::WeighedCriterion main_criterion; - if (path.force_start_index_.has_value()) // Actually handles EZSeamType::USER_SPECIFIED + if (path.force_start_index_.has_value()) // Handles EZSeamType::USER_SPECIFIED with "seam_on_vertex" disabled { // Use a much smaller distance divider because we want points around the forced points to be filtered out very easily constexpr double distance_divider = 1.0; constexpr auto distance_type = DistanceScoringCriterion::DistanceType::Euclidian; main_criterion.criterion = std::make_shared(points, points.at(path.force_start_index_.value()), distance_type, distance_divider); } - else + else if (path.seam_config_.type_ == EZSeamType::SHORTEST || path.seam_config_.type_ == EZSeamType::USER_SPECIFIED) { - if (path.seam_config_.type_ == EZSeamType::SHORTEST) - { - main_criterion.criterion = std::make_shared(points, target_pos); - } - else if ( - path.seam_config_.type_ == EZSeamType::SHARPEST_CORNER - && (path.seam_config_.corner_pref_ != EZSeamCornerPrefType::Z_SEAM_CORNER_PREF_NONE && path.seam_config_.corner_pref_ != EZSeamCornerPrefType::PLUGIN)) - { - main_criterion.criterion = std::make_shared(points, path.seam_config_.corner_pref_); - } - else if (path.seam_config_.type_ == EZSeamType::RANDOM) - { - main_criterion.criterion = std::make_shared(); - } + main_criterion.criterion = std::make_shared(points, target_pos); + } + else if ( + path.seam_config_.type_ == EZSeamType::SHARPEST_CORNER + && (path.seam_config_.corner_pref_ != EZSeamCornerPrefType::Z_SEAM_CORNER_PREF_NONE && path.seam_config_.corner_pref_ != EZSeamCornerPrefType::PLUGIN)) + { + main_criterion.criterion = std::make_shared(points, path.seam_config_.corner_pref_); + } + else if (path.seam_config_.type_ == EZSeamType::RANDOM) + { + main_criterion.criterion = std::make_shared(); } if (main_criterion.criterion) diff --git a/src/FffGcodeWriter.cpp b/src/FffGcodeWriter.cpp index 76b3d1b719..58e72d8019 100644 --- a/src/FffGcodeWriter.cpp +++ b/src/FffGcodeWriter.cpp @@ -1169,6 +1169,7 @@ FffGcodeWriter::ProcessLayerResult FffGcodeWriter::processLayer(const SliceDataS const Scene& scene = Application::getInstance().current_slice_->scene; + coord_t comb_offset_from_outlines = 0; coord_t avoid_distance = 0; // minimal avoid distance is zero const std::vector extruder_is_used = storage.getExtrudersUsed(); for (size_t extruder_nr = 0; extruder_nr < scene.extruders.size(); extruder_nr++) @@ -1181,6 +1182,8 @@ FffGcodeWriter::ProcessLayerResult FffGcodeWriter::processLayer(const SliceDataS { avoid_distance = std::max(avoid_distance, extruder.settings_.get("travel_avoid_distance")); } + + comb_offset_from_outlines = std::max(comb_offset_from_outlines, extruder.settings_.get("retraction_combing_avoid_distance")); } } @@ -1196,7 +1199,6 @@ FffGcodeWriter::ProcessLayerResult FffGcodeWriter::processLayer(const SliceDataS } max_inner_wall_width = std::max(max_inner_wall_width, mesh_inner_wall_width); } - const coord_t comb_offset_from_outlines = max_inner_wall_width * 2; const size_t first_extruder = findUsedExtruderIndex(storage, layer_nr, false); diff --git a/src/LayerPlan.cpp b/src/LayerPlan.cpp index 39f3e1b510..de3594c58d 100644 --- a/src/LayerPlan.cpp +++ b/src/LayerPlan.cpp @@ -196,10 +196,10 @@ Shape LayerPlan::computeCombBoundary(const CombBoundary boundary_type) switch (boundary_type) { case CombBoundary::MINIMUM: - offset = -mesh.settings.get("machine_nozzle_size") / 2 - mesh.settings.get("wall_line_width_0") / 2 - extra_offset; + offset = -(mesh.settings.get("machine_nozzle_size") / 2 + mesh.settings.get("wall_line_width_0") / 2 + extra_offset); break; case CombBoundary::PREFERRED: - offset = -mesh.settings.get("machine_nozzle_size") * 3 / 2 - mesh.settings.get("wall_line_width_0") / 2 - extra_offset; + offset = -(mesh.settings.get("retraction_combing_avoid_distance") + mesh.settings.get("wall_line_width_0") / 2 + extra_offset); break; default: offset = 0; @@ -1322,7 +1322,7 @@ std::vector for (const auto& reversed_chunk : paths | ranges::views::enumerate | ranges::views::reverse | ranges::views::chunk_by( - [](const auto&path_a, const auto&path_b) + [](const auto& path_a, const auto& path_b) { return (! std::get<1>(path_a).isTravelPath()) || std::get<1>(path_b).isTravelPath(); })) @@ -2512,8 +2512,24 @@ void LayerPlan::writeGCode(GCodeExport& gcode) { ExtruderPlan& extruder_plan = extruder_plans_[extruder_plan_idx]; - const RetractionAndWipeConfig* retraction_config - = current_mesh ? ¤t_mesh->retraction_wipe_config : &storage_.retraction_wipe_config_per_extruder[extruder_plan.extruder_nr_]; + auto get_retraction_config = [&extruder_nr, this](std::shared_ptr& mesh) -> std::optional + { + if (mesh) + { + if (extruder_nr == mesh->settings.get("extruder_nr")) [[likely]] + { + return &mesh->retraction_wipe_config; + } + + // We are printing a part of a mesh with a different extruder, use this extruder settings instead (mesh-specific settings will be ignored) + return &storage_.retraction_wipe_config_per_extruder[extruder_nr]; + } + + // We have no mesh yet, a more global config should be used + return std::nullopt; + }; + + const RetractionAndWipeConfig* retraction_config = get_retraction_config(current_mesh).value_or(&storage_.retraction_wipe_config_per_extruder[extruder_plan.extruder_nr_]); coord_t z_hop_height = retraction_config->retraction_config.zHop; if (extruder_nr != extruder_plan.extruder_nr_) @@ -2697,7 +2713,7 @@ void LayerPlan::writeGCode(GCodeExport& gcode) if (path.retract) { - retraction_config = path.mesh ? &path.mesh->retraction_wipe_config : retraction_config; + retraction_config = get_retraction_config(path.mesh).value_or(retraction_config); gcode.writeRetraction(retraction_config->retraction_config); if (path.retract_for_nozzle_switch) {