Skip to content
Snippets Groups Projects
Marlin_main.cpp 226 KiB
Newer Older
  • Learn to ignore specific revisions
  •         } //xProbe
          } //yProbe
    
          #if ENABLED(DEBUG_LEVELING_FEATURE)
            if (marlin_debug_flags & DEBUG_LEVELING) {
              print_xyz("> probing complete > current_position", current_position);
            }
    
          clean_up_after_endstop_move();
    
    
            if (!dryrun) extrapolate_unprobed_bed_level();
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            print_bed_level();
    
    Scott Lahteine's avatar
    Scott Lahteine committed
          #else // !DELTA
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            // solve lsq problem
    
            double plane_equation_coefficients[3];
            qr_solve(plane_equation_coefficients, abl2, 3, eqnAMatrix, eqnBVector);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
    
            mean /= abl2;
    
            if (verbose_level) {
              SERIAL_PROTOCOLPGM("Eqn coefficients: a: ");
              SERIAL_PROTOCOL_F(plane_equation_coefficients[0], 8);
              SERIAL_PROTOCOLPGM(" b: ");
              SERIAL_PROTOCOL_F(plane_equation_coefficients[1], 8);
              SERIAL_PROTOCOLPGM(" d: ");
              SERIAL_PROTOCOL_F(plane_equation_coefficients[2], 8);
    
              SERIAL_EOL;
    
    Scott Lahteine's avatar
    Scott Lahteine committed
              if (verbose_level > 2) {
                SERIAL_PROTOCOLPGM("Mean of sampled points: ");
                SERIAL_PROTOCOL_F(mean, 8);
                SERIAL_EOL;
              }
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            }
    
            if (!dryrun) set_bed_level_equation_lsq(plane_equation_coefficients);
    
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            // Show the Topography map if enabled
            if (do_topography_map) {
    
              SERIAL_PROTOCOLPGM(" \nBed Height Topography: \n");
              SERIAL_PROTOCOLPGM("+-----------+\n");
              SERIAL_PROTOCOLPGM("|...Back....|\n");
              SERIAL_PROTOCOLPGM("|Left..Right|\n");
              SERIAL_PROTOCOLPGM("|...Front...|\n");
              SERIAL_PROTOCOLPGM("+-----------+\n");
    
    
              float min_diff = 999;
    
    Scott Lahteine's avatar
    Scott Lahteine committed
              for (int yy = auto_bed_leveling_grid_points - 1; yy >= 0; yy--) {
                for (int xx = 0; xx < auto_bed_leveling_grid_points; xx++) {
    
    AnHardt's avatar
    AnHardt committed
                  int ind = indexIntoAB[xx][yy];
    
    Scott Lahteine's avatar
    Scott Lahteine committed
                  float diff = eqnBVector[ind] - mean;
    
                  float x_tmp = eqnAMatrix[ind + 0 * abl2],
    
                  apply_rotation_xyz(plan_bed_level_matrix,x_tmp,y_tmp,z_tmp);
    
                  if (eqnBVector[ind] - z_tmp < min_diff)
    
    Scott Lahteine's avatar
    Scott Lahteine committed
                  if (diff >= 0.0)
                    SERIAL_PROTOCOLPGM(" +");   // Include + for column alignment
                  else
    
                    SERIAL_PROTOCOLCHAR(' ');
    
    Scott Lahteine's avatar
    Scott Lahteine committed
                  SERIAL_PROTOCOL_F(diff, 5);
                } // xx
                SERIAL_EOL;
              } // yy
    
              SERIAL_EOL;
    
              if (verbose_level > 3) {
                SERIAL_PROTOCOLPGM(" \nCorrected Bed Height vs. Bed Topology: \n");
    
                for (int yy = auto_bed_leveling_grid_points - 1; yy >= 0; yy--) {
                  for (int xx = 0; xx < auto_bed_leveling_grid_points; xx++) {
    
    AnHardt's avatar
    AnHardt committed
                    int ind = indexIntoAB[xx][yy];
    
                    float x_tmp = eqnAMatrix[ind + 0 * abl2],
                      y_tmp = eqnAMatrix[ind + 1 * abl2],
                      z_tmp = 0;
    
                    apply_rotation_xyz(plan_bed_level_matrix,x_tmp,y_tmp,z_tmp);
    
                    float diff = eqnBVector[ind] - z_tmp - min_diff;
                    if (diff >= 0.0)
                      SERIAL_PROTOCOLPGM(" +");
                    // Include + for column alignment
                    else
                      SERIAL_PROTOCOLCHAR(' ');
                    SERIAL_PROTOCOL_F(diff, 5);
                  } // xx
                  SERIAL_EOL;
                } // yy
    
          #endif //!DELTA
    
    
        #else // !AUTO_BED_LEVELING_GRID
    
    
          #if ENABLED(DEBUG_LEVELING_FEATURE)
            if (marlin_debug_flags & DEBUG_LEVELING) {
              SERIAL_ECHOLNPGM("> 3-point Leveling");
            }
    
    Scott Lahteine's avatar
    Scott Lahteine committed
          // Actions for each probe
    
    Scott Lahteine's avatar
    Scott Lahteine committed
          ProbeAction p1, p2, p3;
    
          if (deploy_probe_for_each_reading)
            p1 = p2 = p3 = ProbeDeployAndStow;
    
    Scott Lahteine's avatar
    Scott Lahteine committed
          else
    
            p1 = ProbeDeploy, p2 = ProbeStay, p3 = ProbeStow;
    
    Scott Lahteine's avatar
    Scott Lahteine committed
    
          // Probe at 3 arbitrary points
          float z_at_pt_1 = probe_pt(ABL_PROBE_PT_1_X, ABL_PROBE_PT_1_Y, Z_RAISE_BEFORE_PROBING, p1, verbose_level),
                z_at_pt_2 = probe_pt(ABL_PROBE_PT_2_X, ABL_PROBE_PT_2_Y, current_position[Z_AXIS] + Z_RAISE_BETWEEN_PROBINGS, p2, verbose_level),
                z_at_pt_3 = probe_pt(ABL_PROBE_PT_3_X, ABL_PROBE_PT_3_Y, current_position[Z_AXIS] + Z_RAISE_BETWEEN_PROBINGS, p3, verbose_level);
    
          clean_up_after_endstop_move();
    
    alexborro's avatar
    alexborro committed
          if (!dryrun) set_bed_level_equation_3pts(z_at_pt_1, z_at_pt_2, z_at_pt_3);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
    
    
        #endif // !AUTO_BED_LEVELING_GRID
    
    Scott Lahteine's avatar
    Scott Lahteine committed
    
    
    Scott Lahteine's avatar
    Scott Lahteine committed
          if (verbose_level > 0)
            plan_bed_level_matrix.debug(" \n\nBed Level Correction Matrix:");
    
    Scott Lahteine's avatar
    Scott Lahteine committed
    
    
          if (!dryrun) {
    
            // Correct the Z height difference from Z probe position and nozzle tip position.
            // The Z height on homing is measured by Z probe, but the Z probe is quite far from the nozzle.
    
            // When the bed is uneven, this height must be corrected.
            float x_tmp = current_position[X_AXIS] + X_PROBE_OFFSET_FROM_EXTRUDER,
                  y_tmp = current_position[Y_AXIS] + Y_PROBE_OFFSET_FROM_EXTRUDER,
                  z_tmp = current_position[Z_AXIS],
    
                  real_z = st_get_position_mm(Z_AXIS);  //get the real Z (since plan_get_position is now correcting the plane)
    
            #if ENABLED(DEBUG_LEVELING_FEATURE)
              if (marlin_debug_flags & DEBUG_LEVELING) {
                SERIAL_ECHOPAIR("> BEFORE apply_rotation_xyz > z_tmp  = ", z_tmp);
                SERIAL_EOL;
                SERIAL_ECHOPAIR("> BEFORE apply_rotation_xyz > real_z = ", real_z);
                SERIAL_EOL;
              }
    
            apply_rotation_xyz(plan_bed_level_matrix, x_tmp, y_tmp, z_tmp); // Apply the correction sending the Z probe offset
    
    
            // Get the current Z position and send it to the planner.
            //
    
            // >> (z_tmp - real_z) : The rotated current Z minus the uncorrected Z (most recent plan_set_position/sync_plan_position)
    
            // >> zprobe_zoffset : Z distance from nozzle to Z probe (set by default, M851, EEPROM, or Menu)
    
            // >> Z_RAISE_AFTER_PROBING : The distance the Z probe will have lifted after the last probe
    
            //
            // >> Should home_offset[Z_AXIS] be included?
            //
            //      Discussion: home_offset[Z_AXIS] was applied in G28 to set the starting Z.
            //      If Z is not tweaked in G29 -and- the Z probe in G29 is not actually "homing" Z...
            //      then perhaps it should not be included here. The purpose of home_offset[] is to
            //      adjust for inaccurate endstops, not for reasonably accurate probes. If it were
            //      added here, it could be seen as a compensating factor for the Z probe.
            //
    
            #if ENABLED(DEBUG_LEVELING_FEATURE)
              if (marlin_debug_flags & DEBUG_LEVELING) {
                SERIAL_ECHOPAIR("> AFTER apply_rotation_xyz > z_tmp  = ", z_tmp);
                SERIAL_EOL;
              }
    
            current_position[Z_AXIS] = -zprobe_zoffset + (z_tmp - real_z)
    
              #if HAS_SERVO_ENDSTOPS || ENABLED(Z_PROBE_ALLEN_KEY) || ENABLED(Z_PROBE_SLED)
    
            // current_position[Z_AXIS] += home_offset[Z_AXIS]; // The Z probe determines Z=0, not "Z home"
    
            sync_plan_position();
    
            #if ENABLED(DEBUG_LEVELING_FEATURE)
              if (marlin_debug_flags & DEBUG_LEVELING) {
                print_xyz("> corrected Z in G29", current_position);
              }
    
          }
        #endif // !DELTA
    
    Scott Lahteine's avatar
    Scott Lahteine committed
    
    
          dock_sled(true); // dock the Z probe
    
        #elif ENABLED(Z_PROBE_ALLEN_KEY) //|| SERVO_LEVELING
    
          stow_z_probe();
    
    Scott Lahteine's avatar
    Scott Lahteine committed
        #endif
    
    Scott Lahteine's avatar
    Scott Lahteine committed
    
    
    Scott Lahteine's avatar
    Scott Lahteine committed
        #ifdef Z_PROBE_END_SCRIPT
    
          #if ENABLED(DEBUG_LEVELING_FEATURE)
            if (marlin_debug_flags & DEBUG_LEVELING) {
              SERIAL_ECHO("Z Probe End Script: ");
              SERIAL_ECHOLNPGM(Z_PROBE_END_SCRIPT);
            }
    
          enqueuecommands_P(PSTR(Z_PROBE_END_SCRIPT));
    
    Scott Lahteine's avatar
    Scott Lahteine committed
          st_synchronize();
        #endif
    
        #if ENABLED(DEBUG_LEVELING_FEATURE)
          if (marlin_debug_flags & DEBUG_LEVELING) {
            SERIAL_ECHOLNPGM("<<< gcode_G29");
          }
    
      #if DISABLED(Z_PROBE_SLED)
    
        /**
         * G30: Do a single Z probe at the current XY
         */
    
        inline void gcode_G30() {
    
          #if HAS_SERVO_ENDSTOPS
            raise_z_for_servo();
          #endif
    
          deploy_z_probe(); // Engage Z Servo endstop if available
    
          st_synchronize();
    
          // TODO: clear the leveling matrix or the planner will be set incorrectly
    
          setup_for_endstop_move();
    
          feedrate = homing_feedrate[Z_AXIS];
    
          SERIAL_PROTOCOLPGM("Bed X: ");
    
          SERIAL_PROTOCOL(current_position[X_AXIS] + 0.0001);
    
          SERIAL_PROTOCOLPGM(" Y: ");
    
          SERIAL_PROTOCOL(current_position[Y_AXIS] + 0.0001);
    
          SERIAL_PROTOCOLPGM(" Z: ");
    
          SERIAL_PROTOCOL(current_position[Z_AXIS] + 0.0001);
          SERIAL_EOL;
    
    
          clean_up_after_endstop_move();
    
    
          #if HAS_SERVO_ENDSTOPS
            raise_z_for_servo();
          #endif
          stow_z_probe(false); // Retract Z Servo endstop if available
    
      #endif //!Z_PROBE_SLED
    
    #endif //AUTO_BED_LEVELING_FEATURE
    
    /**
     * G92: Set current position to given X Y Z E
     */
    inline void gcode_G92() {
      if (!code_seen(axis_codes[E_AXIS]))
        st_synchronize();
    
      for (int i = 0; i < NUM_AXIS; i++) {
    
        if (code_seen(axis_codes[i])) {
    
          float v = current_position[i] = code_value();
    
          if (i == E_AXIS)
    
            plan_set_e_position(v);
    
        #if ENABLED(DELTA) || ENABLED(SCARA)
    
          sync_plan_position_delta();
        #else
          sync_plan_position();
        #endif
      }
    
    bkubicek's avatar
    bkubicek committed
    
    
      /**
       * M0: // M0 - Unconditional stop - Wait for user button press on LCD
       * M1: // M1 - Conditional stop - Wait for user button press on LCD
       */
      inline void gcode_M0_M1() {
    
        char *args = current_command_args;
    
        millis_t codenum = 0;
    
        bool hasP = false, hasS = false;
        if (code_seen('P')) {
    
          codenum = code_value_short(); // milliseconds to wait
    
          hasP = codenum > 0;
        }
        if (code_seen('S')) {
    
          codenum = code_value() * 1000; // seconds to wait
    
          hasS = codenum > 0;
        }
    
    
        if (!hasP && !hasS && *args != '\0')
          lcd_setstatus(args, true);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
        else {
    
          LCD_MESSAGEPGM(MSG_USERWAIT);
    
          #if ENABLED(LCD_PROGRESS_BAR) && PROGRESS_MSG_EXPIRE > 0
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            dontExpireStatus();
          #endif
        }
    
    
        lcd_ignore_click();
        st_synchronize();
    
        refresh_cmd_timeout();
    
        if (codenum > 0) {
    
          codenum += previous_cmd_ms;  // wait until this time for a click
          while (millis() < codenum && !lcd_clicked()) idle();
    
          lcd_ignore_click(false);
        }
        else {
          if (!lcd_detected()) return;
    
          while (!lcd_clicked()) idle();
    
        }
        if (IS_SD_PRINTING)
          LCD_MESSAGEPGM(MSG_RESUMING);
        else
          LCD_MESSAGEPGM(WELCOME_MSG);
      }
    
    #endif // ULTIPANEL
    
    /**
     * M17: Enable power on all stepper motors
     */
    inline void gcode_M17() {
      LCD_MESSAGEPGM(MSG_NO_MOVE);
    
      enable_all_steppers();
    
      /**
       * M20: List SD card to serial output
       */
      inline void gcode_M20() {
        SERIAL_PROTOCOLLNPGM(MSG_BEGIN_FILE_LIST);
        card.ls();
        SERIAL_PROTOCOLLNPGM(MSG_END_FILE_LIST);
      }
    
      /**
       * M21: Init SD Card
       */
      inline void gcode_M21() {
        card.initsd();
      }
    
      /**
       * M22: Release SD Card
       */
      inline void gcode_M22() {
        card.release();
      }
    
      /**
       * M23: Select a file
       */
      inline void gcode_M23() {
    
        card.openFile(current_command_args, true);
    
      /**
       * M24: Start SD Print
       */
      inline void gcode_M24() {
        card.startFileprint();
    
        print_job_start_ms = millis();
    
      /**
       * M25: Pause SD Print
       */
      inline void gcode_M25() {
        card.pauseSDPrint();
      }
    
      /**
       * M26: Set SD Card file index
       */
      inline void gcode_M26() {
        if (card.cardOK && code_seen('S'))
    
          card.setIndex(code_value_short());
    
      /**
       * M27: Get SD Card status
       */
      inline void gcode_M27() {
        card.getStatus();
      }
    
      /**
       * M28: Start SD Write
       */
      inline void gcode_M28() {
    
        card.openFile(current_command_args, false);
    
      /**
       * M29: Stop SD Write
       * Processed in write to file routine above
       */
      inline void gcode_M29() {
        // card.saving = false;
      }
    
      /**
       * M30 <filename>: Delete SD Card file
       */
      inline void gcode_M30() {
        if (card.cardOK) {
          card.closefile();
    
          card.removeFile(current_command_args);
    
    /**
     * M31: Get the time since the start of SD Print (or last M109)
     */
    inline void gcode_M31() {
    
      print_job_stop_ms = millis();
      millis_t t = (print_job_stop_ms - print_job_start_ms) / 1000;
    
      int min = t / 60, sec = t % 60;
      char time[30];
      sprintf_P(time, PSTR("%i min, %i sec"), min, sec);
      SERIAL_ECHO_START;
      SERIAL_ECHOLN(time);
      lcd_setstatus(time);
      autotempShutdown();
    }
    
      /**
       * M32: Select file and start SD Print
       */
      inline void gcode_M32() {
        if (card.sdprinting)
          st_synchronize();
    
        char* namestartpos = strchr(current_command_args, '!');  // Find ! to indicate filename string start.
    
        if (!namestartpos)
    
          namestartpos = current_command_args; // Default name position, 4 letters after the M
    
        else
          namestartpos++; //to skip the '!'
    
        bool call_procedure = code_seen('P') && (seen_pointer < namestartpos);
    
        if (card.cardOK) {
          card.openFile(namestartpos, true, !call_procedure);
    
          if (code_seen('S') && seen_pointer < namestartpos) // "S" (must occur _before_ the filename!)
    
            card.setIndex(code_value_short());
    
          card.startFileprint();
          if (!call_procedure)
    
            print_job_start_ms = millis(); //procedure calls count as normal print time.
    
      #if ENABLED(LONG_FILENAME_HOST_SUPPORT)
    
    
        /**
         * M33: Get the long full path of a file or folder
         *
         * Parameters:
         *   <dospath> Case-insensitive DOS-style path to a file or folder
         *
         * Example:
         *   M33 miscel~1/armchair/armcha~1.gco
         *
         * Output:
         *   /Miscellaneous/Armchair/Armchair.gcode
         */
        inline void gcode_M33() {
    
          card.printLongPath(current_command_args);
    
      /**
       * M928: Start SD Write
       */
      inline void gcode_M928() {
    
        card.openLogFile(current_command_args);
    
    #endif // SDSUPPORT
    
    /**
     * M42: Change pin status via GCode
     */
    inline void gcode_M42() {
      if (code_seen('S')) {
    
        int pin_status = code_value_short(),
    
            pin_number = LED_PIN;
    
        if (code_seen('P') && pin_status >= 0 && pin_status <= 255)
    
          pin_number = code_value_short();
    
        for (uint8_t i = 0; i < COUNT(sensitive_pins); i++) {
    
          if (sensitive_pins[i] == pin_number) {
            pin_number = -1;
            break;
          }
        }
    
        #if HAS_FAN
    
          if (pin_number == FAN_PIN) fanSpeed = pin_status;
        #endif
    
        if (pin_number > -1) {
          pinMode(pin_number, OUTPUT);
          digitalWrite(pin_number, pin_status);
          analogWrite(pin_number, pin_status);
        }
      } // code_seen('S')
    }
    
    #if ENABLED(AUTO_BED_LEVELING_FEATURE) && ENABLED(Z_MIN_PROBE_REPEATABILITY_TEST)
    
      // This is redundant since the SanityCheck.h already checks for a valid Z_MIN_PROBE_PIN, but here for clarity.
      #if ENABLED(Z_MIN_PROBE_ENDSTOP)
    
        #if !HAS_Z_PROBE
    
          #error You must define Z_MIN_PROBE_PIN to enable Z probe repeatability calculation.
    
      #elif !HAS_Z_MIN
    
        #error You must define Z_MIN_PIN to enable Z probe repeatability calculation.
    
       * M48: Z probe repeatability measurement function.
    
       *   M48 <P#> <X#> <Y#> <V#> <E> <L#>
    
    Scott Lahteine's avatar
    Scott Lahteine committed
       *     P = Number of sampled points (4-50, default 10)
    
       *     X = Sample X position
       *     Y = Sample Y position
       *     V = Verbose level (0-4, default=1)
    
       *     E = Engage Z probe for each reading
    
       *     L = Number of legs of movement before probe
    
    Scott Lahteine's avatar
    Scott Lahteine committed
       * This function assumes the bed has been homed.  Specifically, that a G28 command
    
       * as been issued prior to invoking the M48 Z probe repeatability measurement function.
    
       * Any information generated by a prior G29 Bed leveling command will be lost and need to be
       * regenerated.
       */
      inline void gcode_M48() {
    
        double sum = 0.0, mean = 0.0, sigma = 0.0, sample_set[50];
    
        uint8_t verbose_level = 1, n_samples = 10, n_legs = 0;
    
    
        if (code_seen('V')) {
    
          verbose_level = code_value_short();
    
          if (verbose_level < 0 || verbose_level > 4 ) {
            SERIAL_PROTOCOLPGM("?Verbose Level not plausible (0-4).\n");
            return;
          }
        }
    
    Scott Lahteine's avatar
    Scott Lahteine committed
        if (verbose_level > 0)
          SERIAL_PROTOCOLPGM("M48 Z-Probe Repeatability test\n");
    
        if (code_seen('P')) {
    
          n_samples = code_value_short();
    
          if (n_samples < 4 || n_samples > 50) {
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            SERIAL_PROTOCOLPGM("?Sample size not plausible (4-50).\n");
    
    Scott Lahteine's avatar
    Scott Lahteine committed
        double X_current = st_get_position_mm(X_AXIS),
               Y_current = st_get_position_mm(Y_AXIS),
    
    Scott Lahteine's avatar
    Scott Lahteine committed
               Z_current = st_get_position_mm(Z_AXIS),
    
    Scott Lahteine's avatar
    Scott Lahteine committed
               E_current = st_get_position_mm(E_AXIS),
               X_probe_location = X_current, Y_probe_location = Y_current,
               Z_start_location = Z_current + Z_RAISE_BEFORE_PROBING;
    
        bool deploy_probe_for_each_reading = code_seen('E');
    
        if (code_seen('X')) {
    
          X_probe_location = code_value() - X_PROBE_OFFSET_FROM_EXTRUDER;
          if (X_probe_location < X_MIN_POS || X_probe_location > X_MAX_POS) {
    
            out_of_range_error(PSTR("X"));
    
        if (code_seen('Y')) {
    
          Y_probe_location = code_value() -  Y_PROBE_OFFSET_FROM_EXTRUDER;
          if (Y_probe_location < Y_MIN_POS || Y_probe_location > Y_MAX_POS) {
    
            out_of_range_error(PSTR("Y"));
    
        if (code_seen('L')) {
    
          n_legs = code_value_short();
    
          if (n_legs == 1) n_legs = 2;
          if (n_legs < 0 || n_legs > 15) {
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            SERIAL_PROTOCOLPGM("?Number of legs in movement not plausible (0-15).\n");
    
        // Do all the preliminary setup work.   First raise the Z probe.
    
        st_synchronize();
        plan_bed_level_matrix.set_to_identity();
    
    Scott Lahteine's avatar
    Scott Lahteine committed
        plan_buffer_line(X_current, Y_current, Z_start_location, E_current, homing_feedrate[Z_AXIS] / 60, active_extruder);
    
        st_synchronize();
    
        //
        // Now get everything to the specified probe point So we can safely do a probe to
    
        // get us close to the bed.  If the Z-Axis is far from the bed, we don't want to
    
        // use that as a starting point for each probe.
        //
        if (verbose_level > 2)
    
          SERIAL_PROTOCOLPGM("Positioning the probe...\n");
    
    
        plan_buffer_line( X_probe_location, Y_probe_location, Z_start_location,
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            E_current,
    
            homing_feedrate[X_AXIS]/60,
            active_extruder);
        st_synchronize();
    
        current_position[X_AXIS] = X_current = st_get_position_mm(X_AXIS);
        current_position[Y_AXIS] = Y_current = st_get_position_mm(Y_AXIS);
        current_position[Z_AXIS] = Z_current = st_get_position_mm(Z_AXIS);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
        current_position[E_AXIS] = E_current = st_get_position_mm(E_AXIS);
    
        // OK, do the initial probe to get us close to the bed.
    
        // Then retrace the right amount and use that in subsequent probes
        //
    
        deploy_z_probe();
    
        setup_for_endstop_move();
        run_z_probe();
    
        current_position[Z_AXIS] = Z_current = st_get_position_mm(Z_AXIS);
        Z_start_location = st_get_position_mm(Z_AXIS) + Z_RAISE_BEFORE_PROBING;
    
        plan_buffer_line( X_probe_location, Y_probe_location, Z_start_location,
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            E_current,
    
            homing_feedrate[X_AXIS]/60,
            active_extruder);
        st_synchronize();
        current_position[Z_AXIS] = Z_current = st_get_position_mm(Z_AXIS);
    
        if (deploy_probe_for_each_reading) stow_z_probe();
    
        for (uint8_t n=0; n < n_samples; n++) {
    
          // Make sure we are at the probe location
          do_blocking_move_to(X_probe_location, Y_probe_location, Z_start_location); // this also updates current_position
    
            millis_t ms = millis();
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            double radius = ms % (X_MAX_LENGTH / 4),       // limit how far out to go
                   theta = RADIANS(ms % 360L);
            float dir = (ms & 0x0001) ? 1 : -1;            // clockwise or counter clockwise
    
            //SERIAL_ECHOPAIR("starting radius: ",radius);
            //SERIAL_ECHOPAIR("   theta: ",theta);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            //SERIAL_ECHOPAIR("   direction: ",dir);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            //SERIAL_EOL;
    
            for (uint8_t l = 0; l < n_legs - 1; l++) {
    
    Scott Lahteine's avatar
    Scott Lahteine committed
              ms = millis();
              theta += RADIANS(dir * (ms % 20L));
              radius += (ms % 10L) - 5L;
    
              if (radius < 0.0) radius = -radius;
    
              X_current = X_probe_location + cos(theta) * radius;
              X_current = constrain(X_current, X_MIN_POS, X_MAX_POS);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
              Y_current = Y_probe_location + sin(theta) * radius;
    
              Y_current = constrain(Y_current, Y_MIN_POS, Y_MAX_POS);
    
              if (verbose_level > 3) {
                SERIAL_ECHOPAIR("x: ", X_current);
                SERIAL_ECHOPAIR("y: ", Y_current);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
                SERIAL_EOL;
    
              do_blocking_move_to(X_current, Y_current, Z_current); // this also updates current_position
    
    Scott Lahteine's avatar
    Scott Lahteine committed
    
            } // n_legs loop
    
    
            // Go back to the probe location
            do_blocking_move_to(X_probe_location, Y_probe_location, Z_start_location); // this also updates current_position
    
    Scott Lahteine's avatar
    Scott Lahteine committed
    
          } // n_legs
    
          if (deploy_probe_for_each_reading)  {
    
            deploy_z_probe();
    
          setup_for_endstop_move();
          run_z_probe();
    
          sample_set[n] = current_position[Z_AXIS];
    
          //
          // Get the current mean for the data points we have so far
          //
          sum = 0.0;
    
          for (uint8_t j = 0; j <= n; j++) sum += sample_set[j];
    
    Scott Lahteine's avatar
    Scott Lahteine committed
          mean = sum / (n + 1);
    
    
          //
          // Now, use that mean to calculate the standard deviation for the
          // data points we have so far
          //
          sum = 0.0;
    
          for (uint8_t j = 0; j <= n; j++) {
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            float ss = sample_set[j] - mean;
            sum += ss * ss;
          }
          sigma = sqrt(sum / (n + 1));
    
    
          if (verbose_level > 1) {
            SERIAL_PROTOCOL(n+1);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            SERIAL_PROTOCOLPGM(" of ");
    
            SERIAL_PROTOCOL((int)n_samples);
    
            SERIAL_PROTOCOLPGM("   z: ");
            SERIAL_PROTOCOL_F(current_position[Z_AXIS], 6);
    
    Scott Lahteine's avatar
    Scott Lahteine committed
            if (verbose_level > 2) {
              SERIAL_PROTOCOLPGM(" mean: ");
              SERIAL_PROTOCOL_F(mean,6);
              SERIAL_PROTOCOLPGM("   sigma: ");
              SERIAL_PROTOCOL_F(sigma,6);
            }
    
          if (verbose_level > 0) SERIAL_EOL;
    
    Scott Lahteine's avatar
    Scott Lahteine committed
          plan_buffer_line(X_probe_location, Y_probe_location, Z_start_location, current_position[E_AXIS], homing_feedrate[Z_AXIS]/60, active_extruder);
    
          // Stow between
    
          if (deploy_probe_for_each_reading) {
    
            stow_z_probe();
    
        // Stow after
    
        if (!deploy_probe_for_each_reading) {
    
          stow_z_probe();
    
    Scott Lahteine's avatar
    Scott Lahteine committed
          delay(1000);
        }
    
        clean_up_after_endstop_move();
    
        if (verbose_level > 0) {
          SERIAL_PROTOCOLPGM("Mean: ");
          SERIAL_PROTOCOL_F(mean, 6);
    
          SERIAL_EOL;
    
        SERIAL_PROTOCOLPGM("Standard Deviation: ");
        SERIAL_PROTOCOL_F(sigma, 6);
    
        SERIAL_EOL; SERIAL_EOL;
    
    #endif // AUTO_BED_LEVELING_FEATURE && Z_MIN_PROBE_REPEATABILITY_TEST
    
    
    /**
     * M104: Set hot end temperature
     */
    inline void gcode_M104() {
      if (setTargetedHotend(104)) return;
    
      if (marlin_debug_flags & DEBUG_DRYRUN) return;
    
      if (code_seen('S')) {
        float temp = code_value();
        setTargetHotend(temp, target_extruder);
    
        #if ENABLED(DUAL_X_CARRIAGE)
    
          if (dual_x_carriage_mode == DXC_DUPLICATION_MODE && target_extruder == 0)
            setTargetHotend1(temp == 0.0 ? 0.0 : temp + duplicate_extruder_temp_offset);
        #endif
      }
    
    }
    
    /**
     * M105: Read hot end and bed temperature
     */
    inline void gcode_M105() {
      if (setTargetedHotend(105)) return;
    
    
      #if HAS_TEMP_0 || HAS_TEMP_BED || ENABLED(HEATER_0_USES_MAX6675)
    
    Scott Lahteine's avatar
    Scott Lahteine committed
        SERIAL_PROTOCOLPGM(MSG_OK);
    
        #if HAS_TEMP_0 || ENABLED(HEATER_0_USES_MAX6675)
    
          SERIAL_PROTOCOLPGM(" T:");
    
          SERIAL_PROTOCOL_F(degHotend(target_extruder), 1);
    
          SERIAL_PROTOCOLPGM(" /");
    
          SERIAL_PROTOCOL_F(degTargetHotend(target_extruder), 1);
    
        #if HAS_TEMP_BED
    
          SERIAL_PROTOCOLPGM(" B:");
    
          SERIAL_PROTOCOL_F(degBed(), 1);
    
          SERIAL_PROTOCOLPGM(" /");
    
          SERIAL_PROTOCOL_F(degTargetBed(), 1);
    
        for (int8_t e = 0; e < EXTRUDERS; ++e) {
    
          SERIAL_PROTOCOLPGM(" T");
    
          SERIAL_PROTOCOL(e);
    
          SERIAL_PROTOCOLCHAR(':');
    
          SERIAL_PROTOCOL_F(degHotend(e), 1);
    
          SERIAL_PROTOCOLPGM(" /");
    
          SERIAL_PROTOCOL_F(degTargetHotend(e), 1);
    
      #else // !HAS_TEMP_0 && !HAS_TEMP_BED
    
        SERIAL_ERROR_START;
        SERIAL_ERRORLNPGM(MSG_ERR_NO_THERMISTORS);
      #endif
    
      SERIAL_PROTOCOLPGM(" @:");
      #ifdef EXTRUDER_WATTS
    
        SERIAL_PROTOCOL((EXTRUDER_WATTS * getHeaterPower(target_extruder))/127);
        SERIAL_PROTOCOLCHAR('W');
    
        SERIAL_PROTOCOL(getHeaterPower(target_extruder));
    
      #endif
    
      SERIAL_PROTOCOLPGM(" B@:");
      #ifdef BED_WATTS
        SERIAL_PROTOCOL((BED_WATTS * getHeaterPower(-1))/127);
    
        SERIAL_PROTOCOLCHAR('W');
    
      #else
        SERIAL_PROTOCOL(getHeaterPower(-1));
      #endif
    
    
      #if ENABLED(SHOW_TEMP_ADC_VALUES)
    
        #if HAS_TEMP_BED
    
          SERIAL_PROTOCOLPGM("    ADC B:");
          SERIAL_PROTOCOL_F(degBed(),1);
          SERIAL_PROTOCOLPGM("C->");
          SERIAL_PROTOCOL_F(rawBedTemp()/OVERSAMPLENR,0);
    
        for (int8_t cur_extruder = 0; cur_extruder < EXTRUDERS; ++cur_extruder) {
          SERIAL_PROTOCOLPGM("  T");
          SERIAL_PROTOCOL(cur_extruder);
    
          SERIAL_PROTOCOLCHAR(':');
    
          SERIAL_PROTOCOL_F(degHotend(cur_extruder),1);
          SERIAL_PROTOCOLPGM("C->");
          SERIAL_PROTOCOL_F(rawHotendTemp(cur_extruder)/OVERSAMPLENR,0);
    
      SERIAL_EOL;
    
    #if HAS_FAN
    
      /**
       * M106: Set Fan Speed
       */
    
      inline void gcode_M106() { fanSpeed = code_seen('S') ? constrain(code_value_short(), 0, 255) : 255; }
    
      /**
       * M107: Fan Off
       */
      inline void gcode_M107() { fanSpeed = 0; }
    
    #endif // HAS_FAN
    
    /**
     * M109: Wait for extruder(s) to reach temperature
     */
    inline void gcode_M109() {
      if (setTargetedHotend(109)) return;
    
      if (marlin_debug_flags & DEBUG_DRYRUN) return;
    
      LCD_MESSAGEPGM(MSG_HEATING);
    
      no_wait_for_cooling = code_seen('S');
      if (no_wait_for_cooling || code_seen('R')) {
    
        float temp = code_value();
        setTargetHotend(temp, target_extruder);
    
        #if ENABLED(DUAL_X_CARRIAGE)
    
          if (dual_x_carriage_mode == DXC_DUPLICATION_MODE && target_extruder == 0)
            setTargetHotend1(temp == 0.0 ? 0.0 : temp + duplicate_extruder_temp_offset);
    
        autotemp_enabled = code_seen('F');
        if (autotemp_enabled) autotemp_factor = code_value();
        if (code_seen('S')) autotemp_min = code_value();
        if (code_seen('B')) autotemp_max = code_value();
      #endif
    
      millis_t temp_ms = millis();
    
      /* See if we are heating up or cooling down */
    
      target_direction = isHeatingHotend(target_extruder); // true if heating, false if cooling
    
    
      cancel_heatup = false;
    
      #ifdef TEMP_RESIDENCY_TIME
    
        long residency_start_ms = -1;
    
        /* continue to loop until we have reached the target temp
          _and_ until TEMP_RESIDENCY_TIME hasn't passed since we reached it */
    
        while((!cancel_heatup)&&((residency_start_ms == -1) ||
              (residency_start_ms >= 0 && (((unsigned int) (millis() - residency_start_ms)) < (TEMP_RESIDENCY_TIME * 1000UL)))) )
    
        while ( target_direction ? (isHeatingHotend(target_extruder)) : (isCoolingHotend(target_extruder)&&(no_wait_for_cooling==false)) )
    
      #endif //TEMP_RESIDENCY_TIME
    
        { // while loop
    
          if (millis() > temp_ms + 1000UL) { //Print temp & remaining time every 1s while waiting
    
            SERIAL_PROTOCOLPGM("T:");
    
            SERIAL_PROTOCOL_F(degHotend(target_extruder),1);
    
            SERIAL_PROTOCOLPGM(" E:");
    
            SERIAL_PROTOCOL((int)target_extruder);
    
            #ifdef TEMP_RESIDENCY_TIME
              SERIAL_PROTOCOLPGM(" W:");
    
              if (residency_start_ms > -1) {
                temp_ms = ((TEMP_RESIDENCY_TIME * 1000UL) - (millis() - residency_start_ms)) / 1000UL;
                SERIAL_PROTOCOLLN(temp_ms);
    
                SERIAL_PROTOCOLLNPGM("?");
    
              SERIAL_EOL;
    
            temp_ms = millis();
    
          #ifdef TEMP_RESIDENCY_TIME
            // start/restart the TEMP_RESIDENCY_TIME timer whenever we reach target temp for the first time
            // or when current temp falls outside the hysteresis after target temp was reached
    
            if ((residency_start_ms == -1 &&  target_direction && (degHotend(target_extruder) >= (degTargetHotend(target_extruder)-TEMP_WINDOW))) ||
                (residency_start_ms == -1 && !target_direction && (degHotend(target_extruder) <= (degTargetHotend(target_extruder)+TEMP_WINDOW))) ||
                (residency_start_ms > -1 && labs(degHotend(target_extruder) - degTargetHotend(target_extruder)) > TEMP_HYSTERESIS) )
    
              residency_start_ms = millis();
    
            }
          #endif //TEMP_RESIDENCY_TIME
    
      LCD_MESSAGEPGM(MSG_HEATING_COMPLETE);
    
      refresh_cmd_timeout();
    
      print_job_start_ms = previous_cmd_ms;
    
    #if HAS_TEMP_BED
    
    
      /**
       * M190: Sxxx Wait for bed current temp to reach target temp. Waits only when heating
       *       Rxxx Wait for bed current temp to reach target temp. Waits when heating and cooling
       */
      inline void gcode_M190() {
    
        if (marlin_debug_flags & DEBUG_DRYRUN) return;
    
    
        LCD_MESSAGEPGM(MSG_BED_HEATING);
    
        no_wait_for_cooling = code_seen('S');
        if (no_wait_for_cooling || code_seen('R'))
    
          setTargetBed(code_value());
    
    
        millis_t temp_ms = millis();
    
        cancel_heatup = false;
        target_direction = isHeatingBed(); // true if heating, false if cooling
    
    
        while ((target_direction && !cancel_heatup) ? isHeatingBed() : isCoolingBed() && !no_wait_for_cooling) {
    
          millis_t ms = millis();
          if (ms > temp_ms + 1000UL) { //Print Temp Reading every 1 second while heating up.
            temp_ms = ms;
    
            float tt = degHotend(active_extruder);
            SERIAL_PROTOCOLPGM("T:");
            SERIAL_PROTOCOL(tt);
            SERIAL_PROTOCOLPGM(" E:");
            SERIAL_PROTOCOL((int)active_extruder);
            SERIAL_PROTOCOLPGM(" B:");
            SERIAL_PROTOCOL_F(degBed(), 1);
    
            SERIAL_EOL;
    
        LCD_MESSAGEPGM(MSG_BED_DONE);
    
        refresh_cmd_timeout();