surface_bottom_detector.cpp 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. // Jacob Walser: jacob@bluerobotics.com
  2. #include "Sub.h"
  3. // counter to verify contact with bottom
  4. static uint32_t bottom_detector_count = 0;
  5. static uint32_t surface_detector_count = 0;
  6. static float current_depth = 0;
  7. // checks if we have have hit bottom or surface and updates the ap.at_bottom and ap.at_surface flags
  8. // called at MAIN_LOOP_RATE
  9. // ToDo: doesn't need to be called this fast
  10. void Sub::update_surface_and_bottom_detector()
  11. {
  12. static float last_alt = 0.0;//12.20
  13. if (!motors.armed()) { // only update when armed
  14. set_surfaced(false);
  15. set_bottomed(false);
  16. last_alt = barometer.get_altitude();
  17. return;
  18. }
  19. Vector3f velocity;
  20. //ahrs.get_velocity_NED(velocity);//12.20
  21. float cur_alt = barometer.get_altitude(); // m
  22. float vel = AC_AttitudeControl::sqrt_controller(cur_alt-last_alt, pos_control._p_pos_z.kP(), pos_control._accel_z_cms/100, pos_control._dt);
  23. velocity.z = vel;//12.20
  24. static uint16_t i=0;
  25. uint16_t timecnt = (uint16_t)(0.25/MAIN_LOOP_SECONDS);
  26. i++;
  27. if (i>timecnt)
  28. {
  29. i = 0;
  30. last_alt = cur_alt;//12.19
  31. }
  32. // check that we are not moving up or down
  33. bool vel_stationary = velocity.z > -0.05 && velocity.z < 0.05;
  34. if (ap.depth_sensor_present && sensor_health.depth) { // we can use the external pressure sensor for a very accurate and current measure of our z axis position
  35. current_depth = barometer.get_altitude(); // cm
  36. if (ap.at_surface) {
  37. set_surfaced(current_depth > g.surface_depth/100.0 - 0.05); // add a 5cm buffer so it doesn't trigger too often
  38. } else {
  39. set_surfaced(current_depth > g.surface_depth/100.0); // If we are above surface depth, we are surfaced
  40. }
  41. if (motors.limit.throttle_lower && vel_stationary) {
  42. // bottom criteria met - increment the counter and check if we've triggered
  43. if (bottom_detector_count < ((float)BOTTOM_DETECTOR_TRIGGER_SEC)*MAIN_LOOP_RATE) {
  44. bottom_detector_count++;
  45. } else {
  46. set_bottomed(true);
  47. }
  48. } else {
  49. set_bottomed(false);
  50. }
  51. // with no external baro, the only thing we have to go by is a vertical velocity estimate
  52. } else if (vel_stationary) {
  53. if (motors.limit.throttle_upper) {
  54. // surface criteria met, increment counter and see if we've triggered
  55. if (surface_detector_count < ((float)SURFACE_DETECTOR_TRIGGER_SEC)*MAIN_LOOP_RATE) {
  56. surface_detector_count++;
  57. } else {
  58. set_surfaced(true);
  59. }
  60. } else if (motors.limit.throttle_lower) {
  61. // bottom criteria met, increment counter and see if we've triggered
  62. if (bottom_detector_count < ((float)BOTTOM_DETECTOR_TRIGGER_SEC)*MAIN_LOOP_RATE) {
  63. bottom_detector_count++;
  64. } else {
  65. set_bottomed(true);
  66. }
  67. } else { // we're not at the limits of throttle, so reset both detectors
  68. set_surfaced(false);
  69. set_bottomed(false);
  70. }
  71. } else { // we're moving up or down, so reset both detectors
  72. set_surfaced(false);
  73. set_bottomed(false);
  74. }
  75. }
  76. void Sub::set_surfaced(bool at_surface)
  77. {
  78. if (ap.at_surface == at_surface) { // do nothing if state unchanged
  79. return;
  80. }
  81. ap.at_surface = at_surface;
  82. surface_detector_count = 0;
  83. if (ap.at_surface) {
  84. Log_Write_Event(DATA_SURFACED);
  85. } else {
  86. Log_Write_Event(DATA_NOT_SURFACED);
  87. }
  88. }
  89. void Sub::set_bottomed(bool at_bottom)
  90. {
  91. if (ap.at_bottom == at_bottom) { // do nothing if state unchanged
  92. return;
  93. }
  94. ap.at_bottom = at_bottom;
  95. bottom_detector_count = 0;
  96. if (ap.at_bottom) {
  97. Log_Write_Event(DATA_BOTTOMED);
  98. } else {
  99. Log_Write_Event(DATA_NOT_BOTTOMED);
  100. }
  101. }