send, receive, apply motorCommands works (proof of concept); add timeout
- board_control successfully sends motor commands to board_motorctl - board_motorctl receives and applies motor commands note: control pcb currently switches to HTTP mode after startup for testing with data from ui - partially commented in code that has to be reworked - control: send commands via uart instead of to motor objects - board motorctl handled motor: add timeout when no target data received (e.g. control pcb offline / uart bugged) - board motorctl uart: receive motorCommands_t struct and apply data to target state of handled motors - types: fix issue with global motorstateStr variable
This commit is contained in:
		
							parent
							
								
									59a4f8c13f
								
							
						
					
					
						commit
						1e544613ee
					
				| @ -71,7 +71,7 @@ buzzer_t buzzer(GPIO_NUM_12, 100); | |||||||
| httpJoystick httpJoystickMain(configHttpJoystickMain); | httpJoystick httpJoystickMain(configHttpJoystickMain); | ||||||
| 
 | 
 | ||||||
| //create global control object (control.hpp)
 | //create global control object (control.hpp)
 | ||||||
| //controlledArmchair control(configControl, &buzzer, &motorLeft, &motorRight, &joystick, &httpJoystickMain);
 | controlledArmchair control(configControl, &buzzer, &joystick, &httpJoystickMain); | ||||||
| 
 | 
 | ||||||
| //create global automatedArmchair object (for auto-mode) (auto.hpp)
 | //create global automatedArmchair object (for auto-mode) (auto.hpp)
 | ||||||
| automatedArmchair armchair; | automatedArmchair armchair; | ||||||
|  | |||||||
| @ -26,7 +26,7 @@ extern gpio_evaluatedSwitch buttonJoystick; | |||||||
| extern buzzer_t buzzer; | extern buzzer_t buzzer; | ||||||
| 
 | 
 | ||||||
| //create global control object
 | //create global control object
 | ||||||
| //extern controlledArmchair control;
 | extern controlledArmchair control; | ||||||
| 
 | 
 | ||||||
| //create global automatedArmchair object (for auto-mode)
 | //create global automatedArmchair object (for auto-mode)
 | ||||||
| extern automatedArmchair armchair; | extern automatedArmchair armchair; | ||||||
|  | |||||||
| @ -1,3 +1,4 @@ | |||||||
|  | #include "types.hpp" | ||||||
| extern "C" | extern "C" | ||||||
| { | { | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| @ -11,6 +12,7 @@ extern "C" | |||||||
| 
 | 
 | ||||||
| #include "config.hpp" | #include "config.hpp" | ||||||
| #include "control.hpp" | #include "control.hpp" | ||||||
|  | #include "uart.hpp" | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| //used definitions moved from config.hpp:
 | //used definitions moved from config.hpp:
 | ||||||
| @ -23,120 +25,121 @@ const char* controlModeStr[7] = {"IDLE", "JOYSTICK", "MASSAGE", "HTTP", "MQTT", | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| //FIXME controlledMotor class not available for this pcb, rework
 | //FIXME controlledMotor class not available for this pcb, rework
 | ||||||
| //  //-----------------------------
 |   //-----------------------------
 | ||||||
| //  //-------- constructor --------
 |   //-------- constructor --------
 | ||||||
| //  //-----------------------------
 |   //-----------------------------
 | ||||||
| //  controlledArmchair::controlledArmchair (
 |   controlledArmchair::controlledArmchair ( | ||||||
| //          control_config_t config_f,
 |           control_config_t config_f, | ||||||
| //          buzzer_t * buzzer_f,
 |           buzzer_t * buzzer_f, | ||||||
| //          controlledMotor* motorLeft_f,
 |           evaluatedJoystick* joystick_f, | ||||||
| //          controlledMotor* motorRight_f,
 |           httpJoystick* httpJoystick_f | ||||||
| //          evaluatedJoystick* joystick_f,
 |           ){ | ||||||
| //          httpJoystick* httpJoystick_f
 |    | ||||||
| //          ){
 |       //copy configuration
 | ||||||
| //  
 |       config = config_f; | ||||||
| //      //copy configuration
 |       //copy object pointers
 | ||||||
| //      config = config_f;
 |       buzzer = buzzer_f; | ||||||
| //      //copy object pointers
 |       joystick_l = joystick_f, | ||||||
| //      buzzer = buzzer_f;
 |       httpJoystickMain_l = httpJoystick_f; | ||||||
| //      motorLeft = motorLeft_f;
 |       //set default mode from config
 | ||||||
| //      motorRight = motorRight_f;
 |       modePrevious = config.defaultMode; | ||||||
| //      joystick_l = joystick_f,
 |        | ||||||
| //      httpJoystickMain_l = httpJoystick_f;
 |       //TODO declare / configure controlled motors here instead of config (unnecessary that button object is globally available - only used here)?
 | ||||||
| //      //set default mode from config
 |   } | ||||||
| //      modePrevious = config.defaultMode;
 |    | ||||||
| //      
 |    | ||||||
| //      //TODO declare / configure controlled motors here instead of config (unnecessary that button object is globally available - only used here)?
 |    | ||||||
| //  }
 |   //----------------------------------
 | ||||||
| //  
 |   //---------- Handle loop -----------
 | ||||||
| //  
 |   //----------------------------------
 | ||||||
| //  
 |   //function that repeatedly generates motor commands depending on the current mode
 | ||||||
| //  //----------------------------------
 |   //also handles fading and current-limit
 | ||||||
| //  //---------- Handle loop -----------
 |   void controlledArmchair::startHandleLoop() { | ||||||
| //  //----------------------------------
 |       while (1){ | ||||||
| //  //function that repeatedly generates motor commands depending on the current mode
 |           ESP_LOGV(TAG, "control task executing... mode=%s", controlModeStr[(int)mode]); | ||||||
| //  //also handles fading and current-limit
 |    | ||||||
| //  void controlledArmchair::startHandleLoop() {
 |           switch(mode) { | ||||||
| //      while (1){
 |               default: | ||||||
| //          ESP_LOGV(TAG, "control task executing... mode=%s", controlModeStr[(int)mode]);
 |                   mode = controlMode_t::IDLE; | ||||||
| //  
 |                   break; | ||||||
| //          switch(mode) {
 |    | ||||||
| //              default:
 |               case controlMode_t::IDLE: | ||||||
| //                  mode = controlMode_t::IDLE;
 |                   //copy preset commands for idling both motors
 | ||||||
| //                  break;
 |                   commands = cmds_bothMotorsIdle; | ||||||
| //  
 | 				  uart_sendStruct<motorCommands_t>(commands); | ||||||
| //              case controlMode_t::IDLE:
 |                   //motorRight->setTarget(commands.right.state, commands.right.duty); 
 | ||||||
| //                  //copy preset commands for idling both motors
 |                   //motorLeft->setTarget(commands.left.state, commands.left.duty); 
 | ||||||
| //                  commands = cmds_bothMotorsIdle;
 |                   vTaskDelay(200 / portTICK_PERIOD_MS); | ||||||
| //                  motorRight->setTarget(commands.right.state, commands.right.duty); 
 |   #ifdef JOYSTICK_LOG_IN_IDLE | ||||||
| //                  motorLeft->setTarget(commands.left.state, commands.left.duty); 
 |   				//get joystick data here (without using it)
 | ||||||
| //                  vTaskDelay(200 / portTICK_PERIOD_MS);
 |   				//since loglevel is DEBUG, calculateion details is output
 | ||||||
| //  #ifdef JOYSTICK_LOG_IN_IDLE
 |                   joystick_l->getData(); //get joystick data here
 | ||||||
| //  				//get joystick data here (without using it)
 |   #endif | ||||||
| //  				//since loglevel is DEBUG, calculateion details is output
 |                   break; | ||||||
| //                  joystick_l->getData(); //get joystick data here
 |    | ||||||
| //  #endif
 |    | ||||||
| //                  break;
 |               case controlMode_t::JOYSTICK: | ||||||
| //  
 |                   vTaskDelay(20 / portTICK_PERIOD_MS); | ||||||
| //  
 |                   //get current joystick data with getData method of evaluatedJoystick
 | ||||||
| //              case controlMode_t::JOYSTICK:
 |                   stickData = joystick_l->getData(); | ||||||
| //                  vTaskDelay(20 / portTICK_PERIOD_MS);
 |                   //additionaly scale coordinates (more detail in slower area)
 | ||||||
| //                  //get current joystick data with getData method of evaluatedJoystick
 |                   joystick_scaleCoordinatesLinear(&stickData, 0.6, 0.35); //TODO: add scaling parameters to config
 | ||||||
| //                  stickData = joystick_l->getData();
 |                   //generate motor commands
 | ||||||
| //                  //additionaly scale coordinates (more detail in slower area)
 |                   commands = joystick_generateCommandsDriving(stickData, altStickMapping); | ||||||
| //                  joystick_scaleCoordinatesLinear(&stickData, 0.6, 0.35); //TODO: add scaling parameters to config
 |                   //apply motor commands
 | ||||||
| //                  //generate motor commands
 | 				  uart_sendStruct<motorCommands_t>(commands); | ||||||
| //                  commands = joystick_generateCommandsDriving(stickData, altStickMapping);
 |                   //motorRight->setTarget(commands.right.state, commands.right.duty); 
 | ||||||
| //                  //apply motor commands
 |                   //motorLeft->setTarget(commands.left.state, commands.left.duty); 
 | ||||||
| //                  motorRight->setTarget(commands.right.state, commands.right.duty); 
 |                   //TODO make motorctl.setTarget also accept motorcommand struct directly
 | ||||||
| //                  motorLeft->setTarget(commands.left.state, commands.left.duty); 
 |                   break; | ||||||
| //                  //TODO make motorctl.setTarget also accept motorcommand struct directly
 |    | ||||||
| //                  break;
 |    | ||||||
| //  
 |               case controlMode_t::MASSAGE: | ||||||
| //  
 |                   vTaskDelay(10 / portTICK_PERIOD_MS); | ||||||
| //              case controlMode_t::MASSAGE:
 |                   //--- read joystick ---
 | ||||||
| //                  vTaskDelay(10 / portTICK_PERIOD_MS);
 |                   //only update joystick data when input not frozen
 | ||||||
| //                  //--- read joystick ---
 |                   if (!freezeInput){ | ||||||
| //                  //only update joystick data when input not frozen
 |                       stickData = joystick_l->getData(); | ||||||
| //                  if (!freezeInput){
 |                   } | ||||||
| //                      stickData = joystick_l->getData();
 |                   //--- generate motor commands ---
 | ||||||
| //                  }
 |                   //pass joystick data from getData method of evaluatedJoystick to generateCommandsShaking function
 | ||||||
| //                  //--- generate motor commands ---
 |                   commands = joystick_generateCommandsShaking(stickData); | ||||||
| //                  //pass joystick data from getData method of evaluatedJoystick to generateCommandsShaking function
 |                   //apply motor commands
 | ||||||
| //                  commands = joystick_generateCommandsShaking(stickData);
 | 				  uart_sendStruct<motorCommands_t>(commands); | ||||||
| //                  //apply motor commands
 |                   //motorRight->setTarget(commands.right.state, commands.right.duty); 
 | ||||||
| //                  motorRight->setTarget(commands.right.state, commands.right.duty); 
 |                   //motorLeft->setTarget(commands.left.state, commands.left.duty); 
 | ||||||
| //                  motorLeft->setTarget(commands.left.state, commands.left.duty); 
 |                   break; | ||||||
| //                  break;
 |    | ||||||
| //  
 |    | ||||||
| //  
 |               case controlMode_t::HTTP: | ||||||
| //              case controlMode_t::HTTP:
 |                   //--- get joystick data from queue ---
 | ||||||
| //                  //--- get joystick data from queue ---
 |                   //Note this function waits several seconds (httpconfig.timeoutMs) for data to arrive, otherwise Center data or NULL is returned
 | ||||||
| //                  //Note this function waits several seconds (httpconfig.timeoutMs) for data to arrive, otherwise Center data or NULL is returned
 |                   //TODO: as described above, when changing modes it might delay a few seconds for the change to apply
 | ||||||
| //                  //TODO: as described above, when changing modes it might delay a few seconds for the change to apply
 |                   stickData = httpJoystickMain_l->getData(); | ||||||
| //                  stickData = httpJoystickMain_l->getData();
 |                   //scale coordinates additionally (more detail in slower area)
 | ||||||
| //                  //scale coordinates additionally (more detail in slower area)
 |                   joystick_scaleCoordinatesLinear(&stickData, 0.6, 0.4); //TODO: add scaling parameters to config
 | ||||||
| //                  joystick_scaleCoordinatesLinear(&stickData, 0.6, 0.4); //TODO: add scaling parameters to config
 |                   ESP_LOGD(TAG, "generating commands from x=%.3f  y=%.3f  radius=%.3f  angle=%.3f", stickData.x, stickData.y, stickData.radius, stickData.angle); | ||||||
| //                  ESP_LOGD(TAG, "generating commands from x=%.3f  y=%.3f  radius=%.3f  angle=%.3f", stickData.x, stickData.y, stickData.radius, stickData.angle);
 |                   //--- generate motor commands ---
 | ||||||
| //                  //--- generate motor commands ---
 |                   //Note: timeout (no data received) is handled in getData method
 | ||||||
| //                  //Note: timeout (no data received) is handled in getData method
 |                   commands = joystick_generateCommandsDriving(stickData, altStickMapping); | ||||||
| //                  commands = joystick_generateCommandsDriving(stickData, altStickMapping);
 |    | ||||||
| //  
 |                   //--- apply commands to motors ---
 | ||||||
| //                  //--- apply commands to motors ---
 |                   //TODO make motorctl.setTarget also accept motorcommand struct directly
 | ||||||
| //                  //TODO make motorctl.setTarget also accept motorcommand struct directly
 | 				  uart_sendStruct<motorCommands_t>(commands); | ||||||
| //                  motorRight->setTarget(commands.right.state, commands.right.duty); 
 |                   //motorRight->setTarget(commands.right.state, commands.right.duty); 
 | ||||||
| //                  motorLeft->setTarget(commands.left.state, commands.left.duty); 
 |                   //motorLeft->setTarget(commands.left.state, commands.left.duty); 
 | ||||||
| //                 break;
 |                  break; | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //              case controlMode_t::AUTO:
 |               case controlMode_t::AUTO: | ||||||
| //                  vTaskDelay(20 / portTICK_PERIOD_MS);
 |                   vTaskDelay(20 / portTICK_PERIOD_MS); | ||||||
| //                 //generate commands
 | //                 //generate commands
 | ||||||
| //                 commands = armchair.generateCommands(&instruction);
 | //                 commands = armchair.generateCommands(&instruction);
 | ||||||
| //                  //--- apply commands to motors ---
 | //                  //--- apply commands to motors ---
 | ||||||
| //                  //TODO make motorctl.setTarget also accept motorcommand struct directly
 | //                  //TODO make motorctl.setTarget also accept motorcommand struct directly
 | ||||||
| //                 motorRight->setTarget(commands.right.state, commands.right.duty); 
 | //				  uart_sendStruct<motorCommands_t>(commands);
 | ||||||
| //                 motorLeft->setTarget(commands.left.state, commands.left.duty); 
 | //                 //motorRight->setTarget(commands.right.state, commands.right.duty); 
 | ||||||
|  | //                 //motorLeft->setTarget(commands.left.state, commands.left.duty); 
 | ||||||
| //  
 | //  
 | ||||||
| //                 //process received instruction
 | //                 //process received instruction
 | ||||||
| //                 switch (instruction) {
 | //                 switch (instruction) {
 | ||||||
| @ -149,124 +152,125 @@ const char* controlModeStr[7] = {"IDLE", "JOYSTICK", "MASSAGE", "HTTP", "MQTT", | |||||||
| //                         changeMode(controlMode_t::JOYSTICK);
 | //                         changeMode(controlMode_t::JOYSTICK);
 | ||||||
| //                         break;
 | //                         break;
 | ||||||
| //                     case auto_instruction_t::RESET_ACCEL_DECEL:
 | //                     case auto_instruction_t::RESET_ACCEL_DECEL:
 | ||||||
| //                         //enable downfading (set to default value)
 | ////                         //enable downfading (set to default value)
 | ||||||
| //                         motorLeft->setFade(fadeType_t::DECEL, true);
 | ////                         motorLeft->setFade(fadeType_t::DECEL, true);
 | ||||||
| //                         motorRight->setFade(fadeType_t::DECEL, true);
 | ////                         motorRight->setFade(fadeType_t::DECEL, true);
 | ||||||
| //                         //set upfading to default value
 | ////                         //set upfading to default value
 | ||||||
| //                         motorLeft->setFade(fadeType_t::ACCEL, true);
 | ////                         motorLeft->setFade(fadeType_t::ACCEL, true);
 | ||||||
| //                         motorRight->setFade(fadeType_t::ACCEL, true);
 | ////                         motorRight->setFade(fadeType_t::ACCEL, true);
 | ||||||
| //                         break;
 | ////                         break;
 | ||||||
| //                     case auto_instruction_t::RESET_ACCEL:
 | //                     case auto_instruction_t::RESET_ACCEL:
 | ||||||
| //                         //set upfading to default value
 | ////                         //set upfading to default value
 | ||||||
| //                         motorLeft->setFade(fadeType_t::ACCEL, true);
 | ////                         motorLeft->setFade(fadeType_t::ACCEL, true);
 | ||||||
| //                         motorRight->setFade(fadeType_t::ACCEL, true);
 | ////                         motorRight->setFade(fadeType_t::ACCEL, true);
 | ||||||
| //                         break;
 | ////                         break;
 | ||||||
| //                     case auto_instruction_t::RESET_DECEL:
 | //                     case auto_instruction_t::RESET_DECEL:
 | ||||||
| //                         //enable downfading (set to default value)
 | ////                         //enable downfading (set to default value)
 | ||||||
| //                         motorLeft->setFade(fadeType_t::DECEL, true);
 | ////                         motorLeft->setFade(fadeType_t::DECEL, true);
 | ||||||
| //                         motorRight->setFade(fadeType_t::DECEL, true);
 | ////                         motorRight->setFade(fadeType_t::DECEL, true);
 | ||||||
| //                         break;
 | //                         break;
 | ||||||
| //                 }
 | //                 }
 | ||||||
| //                 break;
 |                  break; | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //                //TODO: add other modes here
 |                 //TODO: add other modes here
 | ||||||
| //          }
 |           } | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //          //--- run actions based on received button button event ---
 |           //--- run actions based on received button button event ---
 | ||||||
| //  		//note: buttonCount received by sendButtonEvent method called from button.cpp
 |   		//note: buttonCount received by sendButtonEvent method called from button.cpp
 | ||||||
| //          //TODO: what if variable gets set from other task during this code? -> mutex around this code
 |           //TODO: what if variable gets set from other task during this code? -> mutex around this code
 | ||||||
| //          switch (buttonCount) {
 |           switch (buttonCount) { | ||||||
| //              case 1: //define joystick center or freeze input
 |               case 1: //define joystick center or freeze input
 | ||||||
| //                  if (mode == controlMode_t::JOYSTICK){
 |                   if (mode == controlMode_t::JOYSTICK){ | ||||||
| //                      //joystick mode: calibrate joystick
 |                       //joystick mode: calibrate joystick
 | ||||||
| //                      joystick_l->defineCenter();
 |                       joystick_l->defineCenter(); | ||||||
| //                  } else if (mode == controlMode_t::MASSAGE){
 |                   } else if (mode == controlMode_t::MASSAGE){ | ||||||
| //                      //massage mode: toggle freeze of input (lock joystick at current values)
 |                       //massage mode: toggle freeze of input (lock joystick at current values)
 | ||||||
| //                      freezeInput = !freezeInput;
 |                       freezeInput = !freezeInput; | ||||||
| //                      if (freezeInput){
 |                       if (freezeInput){ | ||||||
| //                          buzzer->beep(5, 40, 25);
 |                           buzzer->beep(5, 40, 25); | ||||||
| //                      } else {
 |                       } else { | ||||||
| //                          buzzer->beep(1, 300, 100);
 |                           buzzer->beep(1, 300, 100); | ||||||
| //                      }
 |                       } | ||||||
| //                  }
 |                   } | ||||||
| //                  break;
 |                   break; | ||||||
| //  
 |    | ||||||
| //              case 12: //toggle alternative joystick mapping (reverse swapped) 
 |               case 12: //toggle alternative joystick mapping (reverse swapped) 
 | ||||||
| //                  altStickMapping = !altStickMapping;
 |                   altStickMapping = !altStickMapping; | ||||||
| //                  if (altStickMapping){
 |                   if (altStickMapping){ | ||||||
| //                      buzzer->beep(6, 70, 50);
 |                       buzzer->beep(6, 70, 50); | ||||||
| //                  } else {
 |                   } else { | ||||||
| //                      buzzer->beep(1, 500, 100);
 |                       buzzer->beep(1, 500, 100); | ||||||
| //                  }
 |                   } | ||||||
| //                  break;
 |                   break; | ||||||
| //          }
 |           } | ||||||
| //          //--- reset button event --- (only one action per run)
 |           //--- reset button event --- (only one action per run)
 | ||||||
| //          if (buttonCount > 0){
 |           if (buttonCount > 0){ | ||||||
| //              ESP_LOGI(TAG, "resetting button event/count");
 |               ESP_LOGI(TAG, "resetting button event/count"); | ||||||
| //              buttonCount = 0;
 |               buttonCount = 0; | ||||||
| //          }
 |           } | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //          //-----------------------
 |           //-----------------------
 | ||||||
| //          //------ slow loop ------
 |           //------ slow loop ------
 | ||||||
| //          //-----------------------
 |           //-----------------------
 | ||||||
| //          //this section is run about every 5s (+500ms)
 |           //this section is run about every 5s (+500ms)
 | ||||||
| //          if (esp_log_timestamp() - timestamp_SlowLoopLastRun > 5000) {
 |           if (esp_log_timestamp() - timestamp_SlowLoopLastRun > 5000) { | ||||||
| //              ESP_LOGV(TAG, "running slow loop... time since last run: %.1fs", (float)(esp_log_timestamp() - timestamp_SlowLoopLastRun)/1000);
 |               ESP_LOGV(TAG, "running slow loop... time since last run: %.1fs", (float)(esp_log_timestamp() - timestamp_SlowLoopLastRun)/1000); | ||||||
| //              timestamp_SlowLoopLastRun = esp_log_timestamp();
 |               timestamp_SlowLoopLastRun = esp_log_timestamp(); | ||||||
| //  
 |    | ||||||
| //              //run function which detects timeout (switch to idle)
 |               //run function which detects timeout (switch to idle)
 | ||||||
| //              handleTimeout();
 |               handleTimeout(); | ||||||
| //          }
 |           } | ||||||
| //  
 |    | ||||||
| //      }//end while(1)
 |       }//end while(1)
 | ||||||
| //  }//end startHandleLoop
 |   }//end startHandleLoop
 | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  //-----------------------------------
 |   //-----------------------------------
 | ||||||
| //  //---------- resetTimeout -----------
 |   //---------- resetTimeout -----------
 | ||||||
| //  //-----------------------------------
 |   //-----------------------------------
 | ||||||
| //  void controlledArmchair::resetTimeout(){
 |   void controlledArmchair::resetTimeout(){ | ||||||
| //      //TODO mutex
 |       //TODO mutex
 | ||||||
| //      timestamp_lastActivity = esp_log_timestamp();
 |       timestamp_lastActivity = esp_log_timestamp(); | ||||||
| //  }
 |   } | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  //------------------------------------
 |   //------------------------------------
 | ||||||
| //  //--------- sendButtonEvent ----------
 |   //--------- sendButtonEvent ----------
 | ||||||
| //  //------------------------------------
 |   //------------------------------------
 | ||||||
| //  void controlledArmchair::sendButtonEvent(uint8_t count){
 |   void controlledArmchair::sendButtonEvent(uint8_t count){ | ||||||
| //      //TODO mutex - if not replaced with queue
 |       //TODO mutex - if not replaced with queue
 | ||||||
| //      ESP_LOGI(TAG, "setting button event");
 |       ESP_LOGI(TAG, "setting button event"); | ||||||
| //      buttonCount = count;
 |       buttonCount = count; | ||||||
| //  }
 |   } | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  //------------------------------------
 |   //------------------------------------
 | ||||||
| //  //---------- handleTimeout -----------
 |   //---------- handleTimeout -----------
 | ||||||
| //  //------------------------------------
 |   //------------------------------------
 | ||||||
| //  //percentage the duty can vary since last timeout check and still counts as incative 
 |   //percentage the duty can vary since last timeout check and still counts as incative 
 | ||||||
| //  //TODO: add this to config
 |   //TODO: add this to config
 | ||||||
| //  float inactivityTolerance = 10; 
 |   float inactivityTolerance = 10;  | ||||||
| //  
 |    | ||||||
| //  //local function that checks whether two values differ more than a given tolerance
 |   //local function that checks whether two values differ more than a given tolerance
 | ||||||
| //  bool validateActivity(float dutyOld, float dutyNow, float tolerance){
 |   bool validateActivity(float dutyOld, float dutyNow, float tolerance){ | ||||||
| //      float dutyDelta = dutyNow - dutyOld;
 |       float dutyDelta = dutyNow - dutyOld; | ||||||
| //      if (fabs(dutyDelta) < tolerance) {
 |       if (fabs(dutyDelta) < tolerance) { | ||||||
| //          return false; //no significant activity detected
 |           return false; //no significant activity detected
 | ||||||
| //      } else {
 |       } else { | ||||||
| //          return true; //there was activity
 |           return true; //there was activity
 | ||||||
| //      }
 |       } | ||||||
| //  }
 |   } | ||||||
| //  
 |    | ||||||
| //  //function that evaluates whether there is no activity/change on the motor duty for a certain time. If so, a switch to IDLE is issued. - has to be run repeatedly in a slow interval
 |   //function that evaluates whether there is no activity/change on the motor duty for a certain time. If so, a switch to IDLE is issued. - has to be run repeatedly in a slow interval
 | ||||||
| //  void controlledArmchair::handleTimeout(){
 |   //FIXME rework timout to work via uart duy or control input only
 | ||||||
|  |   void controlledArmchair::handleTimeout(){ | ||||||
| //      //check for timeout only when not idling already
 | //      //check for timeout only when not idling already
 | ||||||
| //      if (mode != controlMode_t::IDLE) {
 | //      if (mode != controlMode_t::IDLE) {
 | ||||||
| //          //get current duty from controlled motor objects
 | //          //get current duty from controlled motor objects
 | ||||||
| @ -293,43 +297,44 @@ const char* controlModeStr[7] = {"IDLE", "JOYSTICK", "MASSAGE", "HTTP", "MQTT", | |||||||
| //              ESP_LOGD(TAG, "timeout check: [inactive], last activity %.1f s ago, timeout after %d s", (float)(esp_log_timestamp() - timestamp_lastActivity)/1000, config.timeoutMs/1000);
 | //              ESP_LOGD(TAG, "timeout check: [inactive], last activity %.1f s ago, timeout after %d s", (float)(esp_log_timestamp() - timestamp_lastActivity)/1000, config.timeoutMs/1000);
 | ||||||
| //          }
 | //          }
 | ||||||
| //      }
 | //      }
 | ||||||
| //  }
 |   } | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  //-----------------------------------
 |   //-----------------------------------
 | ||||||
| //  //----------- changeMode ------------
 |   //----------- changeMode ------------
 | ||||||
| //  //-----------------------------------
 |   //-----------------------------------
 | ||||||
| //  //function to change to a specified control mode
 |   //function to change to a specified control mode
 | ||||||
| //  void controlledArmchair::changeMode(controlMode_t modeNew) {
 |   //FIXME FIXME: replace change with motorLeft object with update config via uart
 | ||||||
| //      //reset timeout timer
 |   void controlledArmchair::changeMode(controlMode_t modeNew) { | ||||||
| //      resetTimeout();
 |       //reset timeout timer
 | ||||||
| //  
 |       resetTimeout(); | ||||||
| //      //exit if target mode is already active
 |    | ||||||
| //      if (mode == modeNew) {
 |       //exit if target mode is already active
 | ||||||
| //          ESP_LOGE(TAG, "changeMode: Already in target mode '%s' -> nothing to change", controlModeStr[(int)mode]);
 |       if (mode == modeNew) { | ||||||
| //          return;
 |           ESP_LOGE(TAG, "changeMode: Already in target mode '%s' -> nothing to change", controlModeStr[(int)mode]); | ||||||
| //      }
 |           return; | ||||||
| //  
 |       } | ||||||
| //      //copy previous mode
 |    | ||||||
| //      modePrevious = mode;
 |       //copy previous mode
 | ||||||
| //  
 |       modePrevious = mode; | ||||||
| //  	ESP_LOGW(TAG, "=== changing mode from %s to %s ===", controlModeStr[(int)mode], controlModeStr[(int)modeNew]);
 |    | ||||||
| //  
 |   	ESP_LOGW(TAG, "=== changing mode from %s to %s ===", controlModeStr[(int)mode], controlModeStr[(int)modeNew]); | ||||||
| //  	//========== commands change FROM mode ==========
 |    | ||||||
| //  	//run functions when changing FROM certain mode
 |   	//========== commands change FROM mode ==========
 | ||||||
| //  	switch(modePrevious){
 |   	//run functions when changing FROM certain mode
 | ||||||
| //  		default:
 |   	switch(modePrevious){ | ||||||
| //  			ESP_LOGI(TAG, "noting to execute when changing FROM this mode");
 |   		default: | ||||||
| //  			break;
 |   			ESP_LOGI(TAG, "noting to execute when changing FROM this mode"); | ||||||
| //  
 |   			break; | ||||||
| //  #ifdef JOYSTICK_LOG_IN_IDLE
 |    | ||||||
| //  		case controlMode_t::IDLE:
 |   #ifdef JOYSTICK_LOG_IN_IDLE | ||||||
| //  			ESP_LOGI(TAG, "disabling debug output for 'evaluatedJoystick'");
 |   		case controlMode_t::IDLE: | ||||||
| //  			esp_log_level_set("evaluatedJoystick", ESP_LOG_WARN); //FIXME: loglevel from config
 |   			ESP_LOGI(TAG, "disabling debug output for 'evaluatedJoystick'"); | ||||||
| //  			break;
 |   			esp_log_level_set("evaluatedJoystick", ESP_LOG_WARN); //FIXME: loglevel from config
 | ||||||
| //  #endif
 |   			break; | ||||||
| //  
 |   #endif | ||||||
|  |    | ||||||
| //  		case controlMode_t::HTTP:
 | //  		case controlMode_t::HTTP:
 | ||||||
| //  			ESP_LOGW(TAG, "switching from http mode -> disabling http and wifi");
 | //  			ESP_LOGW(TAG, "switching from http mode -> disabling http and wifi");
 | ||||||
| //  			//stop http server
 | //  			//stop http server
 | ||||||
| @ -368,43 +373,43 @@ const char* controlModeStr[7] = {"IDLE", "JOYSTICK", "MASSAGE", "HTTP", "MQTT", | |||||||
| //              motorLeft->setFade(fadeType_t::ACCEL, true);
 | //              motorLeft->setFade(fadeType_t::ACCEL, true);
 | ||||||
| //              motorRight->setFade(fadeType_t::ACCEL, true);
 | //              motorRight->setFade(fadeType_t::ACCEL, true);
 | ||||||
| //              break;
 | //              break;
 | ||||||
| //      }
 |       } | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //      //========== commands change TO mode ==========
 |       //========== commands change TO mode ==========
 | ||||||
| //      //run functions when changing TO certain mode
 |       //run functions when changing TO certain mode
 | ||||||
| //      switch(modeNew){
 |       switch(modeNew){ | ||||||
| //          default:
 |           default: | ||||||
| //              ESP_LOGI(TAG, "noting to execute when changing TO this mode");
 |               ESP_LOGI(TAG, "noting to execute when changing TO this mode"); | ||||||
| //              break;
 |               break; | ||||||
| //  
 |    | ||||||
| //  		case controlMode_t::IDLE:
 |   		case controlMode_t::IDLE: | ||||||
| //  			buzzer->beep(1, 1500, 0);
 |   			buzzer->beep(1, 1500, 0); | ||||||
| //  #ifdef JOYSTICK_LOG_IN_IDLE
 |   #ifdef JOYSTICK_LOG_IN_IDLE | ||||||
| //  			esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG);
 |   			esp_log_level_set("evaluatedJoystick", ESP_LOG_DEBUG); | ||||||
| //  #endif
 |   #endif | ||||||
| //  			break;
 |   			break; | ||||||
| //  
 |    | ||||||
| //          case controlMode_t::HTTP:
 |           case controlMode_t::HTTP: | ||||||
| //              ESP_LOGW(TAG, "switching to http mode -> enabling http and wifi");
 |               ESP_LOGW(TAG, "switching to http mode -> enabling http and wifi"); | ||||||
| //              //start wifi
 |               //start wifi
 | ||||||
| //              //TODO: decide wether ap or client should be started
 |               //TODO: decide wether ap or client should be started
 | ||||||
| //              ESP_LOGI(TAG, "init wifi...");
 |               ESP_LOGI(TAG, "init wifi..."); | ||||||
| //  
 |    | ||||||
| //              //FIXME: make wifi function work here - currently starting wifi at startup (see notes main.cpp)
 |               //FIXME: make wifi function work here - currently starting wifi at startup (see notes main.cpp)
 | ||||||
| //              //wifi_init_client();
 |               //wifi_init_client();
 | ||||||
| //              //wifi_init_ap();
 |               //wifi_init_ap();
 | ||||||
| //  
 |    | ||||||
| //              //wait for wifi
 |               //wait for wifi
 | ||||||
| //              //ESP_LOGI(TAG, "waiting for wifi...");
 |               //ESP_LOGI(TAG, "waiting for wifi...");
 | ||||||
| //              //vTaskDelay(1000 / portTICK_PERIOD_MS);
 |               //vTaskDelay(1000 / portTICK_PERIOD_MS);
 | ||||||
| //  
 |    | ||||||
| //              //start http server
 |               //start http server
 | ||||||
| //              ESP_LOGI(TAG, "init http server...");
 |               ESP_LOGI(TAG, "init http server..."); | ||||||
| //              http_init_server();
 |               http_init_server(); | ||||||
| //              ESP_LOGI(TAG, "done initializing http mode");
 |               ESP_LOGI(TAG, "done initializing http mode"); | ||||||
| //              break;
 |               break; | ||||||
| //  
 |    | ||||||
| //          case controlMode_t::MASSAGE:
 | //          case controlMode_t::MASSAGE:
 | ||||||
| //              ESP_LOGW(TAG, "switching to MASSAGE mode -> reducing fading");
 | //              ESP_LOGW(TAG, "switching to MASSAGE mode -> reducing fading");
 | ||||||
| //              uint32_t shake_msFadeAccel = 500; //TODO: move this to config
 | //              uint32_t shake_msFadeAccel = 500; //TODO: move this to config
 | ||||||
| @ -416,65 +421,65 @@ const char* controlModeStr[7] = {"IDLE", "JOYSTICK", "MASSAGE", "HTTP", "MQTT", | |||||||
| //              motorLeft->setFade(fadeType_t::ACCEL, shake_msFadeAccel);
 | //              motorLeft->setFade(fadeType_t::ACCEL, shake_msFadeAccel);
 | ||||||
| //              motorRight->setFade(fadeType_t::ACCEL, shake_msFadeAccel);
 | //              motorRight->setFade(fadeType_t::ACCEL, shake_msFadeAccel);
 | ||||||
| //              break;
 | //              break;
 | ||||||
| //  
 |    | ||||||
| //      }
 |       } | ||||||
| //  
 |    | ||||||
| //      //--- update mode to new mode ---
 |       //--- update mode to new mode ---
 | ||||||
| //      //TODO: add mutex
 |       //TODO: add mutex
 | ||||||
| //      mode = modeNew;
 |       mode = modeNew; | ||||||
| //  }
 |   } | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  //TODO simplify the following 3 functions? can be replaced by one?
 |   //TODO simplify the following 3 functions? can be replaced by one?
 | ||||||
| //  
 |    | ||||||
| //  //-----------------------------------
 |   //-----------------------------------
 | ||||||
| //  //----------- toggleIdle ------------
 |   //----------- toggleIdle ------------
 | ||||||
| //  //-----------------------------------
 |   //-----------------------------------
 | ||||||
| //  //function to toggle between IDLE and previous active mode
 |   //function to toggle between IDLE and previous active mode
 | ||||||
| //  void controlledArmchair::toggleIdle() {
 |   void controlledArmchair::toggleIdle() { | ||||||
| //      //toggle between IDLE and previous mode
 |       //toggle between IDLE and previous mode
 | ||||||
| //      toggleMode(controlMode_t::IDLE);
 |       toggleMode(controlMode_t::IDLE); | ||||||
| //  }
 |   } | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  //------------------------------------
 |   //------------------------------------
 | ||||||
| //  //----------- toggleModes ------------
 |   //----------- toggleModes ------------
 | ||||||
| //  //------------------------------------
 |   //------------------------------------
 | ||||||
| //  //function to toggle between two modes, but prefer first argument if entirely different mode is currently active
 |   //function to toggle between two modes, but prefer first argument if entirely different mode is currently active
 | ||||||
| //  void controlledArmchair::toggleModes(controlMode_t modePrimary, controlMode_t modeSecondary) {
 |   void controlledArmchair::toggleModes(controlMode_t modePrimary, controlMode_t modeSecondary) { | ||||||
| //      //switch to secondary mode when primary is already active
 |       //switch to secondary mode when primary is already active
 | ||||||
| //      if (mode == modePrimary){
 |       if (mode == modePrimary){ | ||||||
| //          ESP_LOGW(TAG, "toggleModes: switching from primaryMode %s to secondarMode %s", controlModeStr[(int)mode], controlModeStr[(int)modeSecondary]);
 |           ESP_LOGW(TAG, "toggleModes: switching from primaryMode %s to secondarMode %s", controlModeStr[(int)mode], controlModeStr[(int)modeSecondary]); | ||||||
| //          buzzer->beep(2,200,100);
 |           buzzer->beep(2,200,100); | ||||||
| //          changeMode(modeSecondary); //switch to secondary mode
 |           changeMode(modeSecondary); //switch to secondary mode
 | ||||||
| //      } 
 |       }  | ||||||
| //      //switch to primary mode when any other mode is active
 |       //switch to primary mode when any other mode is active
 | ||||||
| //      else {
 |       else { | ||||||
| //          ESP_LOGW(TAG, "toggleModes: switching from %s to primary mode %s", controlModeStr[(int)mode], controlModeStr[(int)modePrimary]);
 |           ESP_LOGW(TAG, "toggleModes: switching from %s to primary mode %s", controlModeStr[(int)mode], controlModeStr[(int)modePrimary]); | ||||||
| //          buzzer->beep(4,200,100);
 |           buzzer->beep(4,200,100); | ||||||
| //          changeMode(modePrimary);
 |           changeMode(modePrimary); | ||||||
| //      }
 |       } | ||||||
| //  }
 |   } | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  
 |    | ||||||
| //  //-----------------------------------
 |   //-----------------------------------
 | ||||||
| //  //----------- toggleMode ------------
 |   //----------- toggleMode ------------
 | ||||||
| //  //-----------------------------------
 |   //-----------------------------------
 | ||||||
| //  //function that toggles between certain mode and previous mode
 |   //function that toggles between certain mode and previous mode
 | ||||||
| //  void controlledArmchair::toggleMode(controlMode_t modePrimary){
 |   void controlledArmchair::toggleMode(controlMode_t modePrimary){ | ||||||
| //  
 |    | ||||||
| //      //switch to previous mode when primary is already active
 |       //switch to previous mode when primary is already active
 | ||||||
| //      if (mode == modePrimary){
 |       if (mode == modePrimary){ | ||||||
| //          ESP_LOGW(TAG, "toggleMode: switching from primaryMode %s to previousMode %s", controlModeStr[(int)mode], controlModeStr[(int)modePrevious]);
 |           ESP_LOGW(TAG, "toggleMode: switching from primaryMode %s to previousMode %s", controlModeStr[(int)mode], controlModeStr[(int)modePrevious]); | ||||||
| //          //buzzer->beep(2,200,100);
 |           //buzzer->beep(2,200,100);
 | ||||||
| //          changeMode(modePrevious); //switch to previous mode
 |           changeMode(modePrevious); //switch to previous mode
 | ||||||
| //      } 
 |       }  | ||||||
| //      //switch to primary mode when any other mode is active
 |       //switch to primary mode when any other mode is active
 | ||||||
| //      else {
 |       else { | ||||||
| //          ESP_LOGW(TAG, "toggleModes: switching from %s to primary mode %s", controlModeStr[(int)mode], controlModeStr[(int)modePrimary]);
 |           ESP_LOGW(TAG, "toggleModes: switching from %s to primary mode %s", controlModeStr[(int)mode], controlModeStr[(int)modePrimary]); | ||||||
| //          //buzzer->beep(4,200,100);
 |           //buzzer->beep(4,200,100);
 | ||||||
| //          changeMode(modePrimary);
 |           changeMode(modePrimary); | ||||||
| //      }
 |       } | ||||||
| //  }
 |   } | ||||||
|  | |||||||
| @ -27,104 +27,100 @@ | |||||||
|    |    | ||||||
|    |    | ||||||
|    |    | ||||||
| //  //==================================
 |   //==================================
 | ||||||
| //  //========= control class ==========
 |   //========= control class ==========
 | ||||||
| //  //==================================
 |   //==================================
 | ||||||
| //  //controls the mode the armchair operates
 |   //controls the mode the armchair operates
 | ||||||
| //  //repeatedly generates the motor commands corresponding to current mode and sends those to motorcontrol
 |   //repeatedly generates the motor commands corresponding to current mode and sends those to motorcontrol
 | ||||||
| //  class controlledArmchair {
 |   class controlledArmchair { | ||||||
| //      public:
 |       public: | ||||||
| //          //--- constructor ---
 |           //--- constructor ---
 | ||||||
| //          controlledArmchair (
 |           controlledArmchair ( | ||||||
| //                  control_config_t config_f,
 |                   control_config_t config_f, | ||||||
| //                  buzzer_t* buzzer_f,
 |                   buzzer_t* buzzer_f, | ||||||
| //                  controlledMotor* motorLeft_f,
 |                   evaluatedJoystick* joystick_f, | ||||||
| //                  controlledMotor* motorRight_f,
 |                   httpJoystick* httpJoystick_f | ||||||
| //                  evaluatedJoystick* joystick_f,
 |                   ); | ||||||
| //                  httpJoystick* httpJoystick_f
 |    | ||||||
| //                  );
 |           //--- functions ---
 | ||||||
| //  
 |           //task that repeatedly generates motor commands depending on the current mode
 | ||||||
| //          //--- functions ---
 |           void startHandleLoop(); | ||||||
| //          //task that repeatedly generates motor commands depending on the current mode
 |    | ||||||
| //          void startHandleLoop();
 |           //function that changes to a specified control mode
 | ||||||
| //  
 |           void changeMode(controlMode_t modeNew); | ||||||
| //          //function that changes to a specified control mode
 |    | ||||||
| //          void changeMode(controlMode_t modeNew);
 |           //function that toggle between IDLE and previous active mode (or default if not switched to certain mode yet)
 | ||||||
| //  
 |           void toggleIdle(); | ||||||
| //          //function that toggle between IDLE and previous active mode (or default if not switched to certain mode yet)
 |    | ||||||
| //          void toggleIdle();
 |           //function that toggles between two modes, but prefers first argument if entirely different mode is currently active
 | ||||||
| //  
 |           void toggleModes(controlMode_t modePrimary, controlMode_t modeSecondary); | ||||||
| //          //function that toggles between two modes, but prefers first argument if entirely different mode is currently active
 |    | ||||||
| //          void toggleModes(controlMode_t modePrimary, controlMode_t modeSecondary);
 |           //toggle between certain mode and previous mode
 | ||||||
| //  
 |           void toggleMode(controlMode_t modePrimary); | ||||||
| //          //toggle between certain mode and previous mode
 |    | ||||||
| //          void toggleMode(controlMode_t modePrimary);
 |           //function that restarts timer which initiates the automatic timeout (switch to IDLE) after certain time of inactivity
 | ||||||
| //  
 |           void resetTimeout(); | ||||||
| //          //function that restarts timer which initiates the automatic timeout (switch to IDLE) after certain time of inactivity
 |    | ||||||
| //          void resetTimeout();
 |           //function for sending a button event (e.g. from button task at event) to control task
 | ||||||
| //  
 |           //TODO: use queue instead?
 | ||||||
| //          //function for sending a button event (e.g. from button task at event) to control task
 |           void sendButtonEvent(uint8_t count); | ||||||
| //          //TODO: use queue instead?
 |    | ||||||
| //          void sendButtonEvent(uint8_t count);
 |       private: | ||||||
| //  
 |    | ||||||
| //      private:
 |           //--- functions ---
 | ||||||
| //  
 |           //function that evaluates whether there is no activity/change on the motor duty for a certain time, if so a switch to IDLE is issued. - has to be run repeatedly in a slow interval
 | ||||||
| //          //--- functions ---
 |           void handleTimeout(); | ||||||
| //          //function that evaluates whether there is no activity/change on the motor duty for a certain time, if so a switch to IDLE is issued. - has to be run repeatedly in a slow interval
 |    | ||||||
| //          void handleTimeout();
 |           //--- objects ---
 | ||||||
| //  
 |           buzzer_t* buzzer; | ||||||
| //          //--- objects ---
 |           httpJoystick* httpJoystickMain_l; | ||||||
| //          buzzer_t* buzzer;
 |           evaluatedJoystick* joystick_l; | ||||||
| //          controlledMotor* motorLeft;
 |    | ||||||
| //          controlledMotor* motorRight;
 |           //---variables ---
 | ||||||
| //          httpJoystick* httpJoystickMain_l;
 |           //struct for motor commands returned by generate functions of each mode
 | ||||||
| //          evaluatedJoystick* joystick_l;
 |           motorCommands_t commands; | ||||||
| //  
 |           //struct with config parameters
 | ||||||
| //          //---variables ---
 |           control_config_t config; | ||||||
| //          //struct for motor commands returned by generate functions of each mode
 |    | ||||||
| //          motorCommands_t commands;
 |           //store joystick data
 | ||||||
| //          //struct with config parameters
 |           joystickData_t stickData; | ||||||
| //          control_config_t config;
 |           bool altStickMapping; //alternative joystick mapping (reverse mapped differently)
 | ||||||
| //  
 |    | ||||||
| //          //store joystick data
 |           //variables for http mode
 | ||||||
| //          joystickData_t stickData;
 |           uint32_t http_timestamp_lastData = 0; | ||||||
| //          bool altStickMapping; //alternative joystick mapping (reverse mapped differently)
 |    | ||||||
| //  
 |           //variables for MASSAGE mode
 | ||||||
| //          //variables for http mode
 |           bool freezeInput = false; | ||||||
| //          uint32_t http_timestamp_lastData = 0;
 |    | ||||||
| //  
 |           //variables for AUTO mode
 | ||||||
| //          //variables for MASSAGE mode
 |           auto_instruction_t instruction = auto_instruction_t::NONE; //variable to receive instructions from automatedArmchair
 | ||||||
| //          bool freezeInput = false;
 |            | ||||||
| //  
 |           //variable to store button event
 | ||||||
| //          //variables for AUTO mode
 |           uint8_t buttonCount = 0; | ||||||
| //          auto_instruction_t instruction = auto_instruction_t::NONE; //variable to receive instructions from automatedArmchair
 |    | ||||||
| //          
 |           //definition of mode enum
 | ||||||
| //          //variable to store button event
 |           controlMode_t mode = controlMode_t::IDLE; | ||||||
| //          uint8_t buttonCount = 0;
 |    | ||||||
| //  
 |           //variable to store mode when toggling IDLE mode 
 | ||||||
| //          //definition of mode enum
 |           controlMode_t modePrevious; //default mode
 | ||||||
| //          controlMode_t mode = controlMode_t::IDLE;
 |    | ||||||
| //  
 |           //command preset for idling motors
 | ||||||
| //          //variable to store mode when toggling IDLE mode 
 |           const motorCommand_t cmd_motorIdle = { | ||||||
| //          controlMode_t modePrevious; //default mode
 |               .state = motorstate_t::IDLE, | ||||||
| //  
 |               .duty = 0 | ||||||
| //          //command preset for idling motors
 |           }; | ||||||
| //          const motorCommand_t cmd_motorIdle = {
 |           const motorCommands_t cmds_bothMotorsIdle = { | ||||||
| //              .state = motorstate_t::IDLE,
 |               .left = cmd_motorIdle, | ||||||
| //              .duty = 0
 |               .right = cmd_motorIdle | ||||||
| //          };
 |           }; | ||||||
| //          const motorCommands_t cmds_bothMotorsIdle = {
 |    | ||||||
| //              .left = cmd_motorIdle,
 |           //variable for slow loop
 | ||||||
| //              .right = cmd_motorIdle
 |           uint32_t timestamp_SlowLoopLastRun = 0; | ||||||
| //          };
 |    | ||||||
| //  
 |           //variables for detecting timeout (switch to idle, after inactivity)
 | ||||||
| //          //variable for slow loop
 |           float dutyLeft_lastActivity = 0; | ||||||
| //          uint32_t timestamp_SlowLoopLastRun = 0;
 |           float dutyRight_lastActivity = 0; | ||||||
| //  
 |           uint32_t timestamp_lastActivity = 0; | ||||||
| //          //variables for detecting timeout (switch to idle, after inactivity)
 |   }; | ||||||
| //          float dutyLeft_lastActivity = 0;
 |    | ||||||
| //          float dutyRight_lastActivity = 0;
 |    | ||||||
| //          uint32_t timestamp_lastActivity = 0;
 |  | ||||||
| //  };
 |  | ||||||
| //  
 |  | ||||||
| //  
 |  | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ extern "C" | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	//custom C files
 | 	//custom C files
 | ||||||
| 	//#include "wifi.h"
 | 	#include "wifi.h" | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -25,7 +25,7 @@ extern "C" | |||||||
| //=========================
 | //=========================
 | ||||||
| //only run uart test code at the end
 | //only run uart test code at the end
 | ||||||
| //disables other functionality
 | //disables other functionality
 | ||||||
| #define UART_TEST_ONLY | //#define UART_TEST_ONLY
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| //tag for logging
 | //tag for logging
 | ||||||
| @ -71,13 +71,13 @@ void task_control( void * pvParameters ){ | |||||||
| //============ button task =============
 | //============ button task =============
 | ||||||
| //======================================
 | //======================================
 | ||||||
| //task that handles the button interface/commands
 | //task that handles the button interface/commands
 | ||||||
| void task_button( void * pvParameters ){ | //void task_button( void * pvParameters ){
 | ||||||
| 	ESP_LOGI(TAG, "Initializing command-button and starting handle loop"); | //	ESP_LOGI(TAG, "Initializing command-button and starting handle loop");
 | ||||||
| 	//create button instance
 | //	//create button instance
 | ||||||
| 	buttonCommands commandButton(&buttonJoystick, &joystick, &control, &buzzer, &motorLeft, &motorRight); | //	buttonCommands commandButton(&buttonJoystick, &joystick, &control, &buzzer, &motorLeft, &motorRight);
 | ||||||
| 	//start handle loop
 | //	//start handle loop
 | ||||||
| 	commandButton.startHandleLoop(); | //	commandButton.startHandleLoop();
 | ||||||
| } | //}
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -85,16 +85,16 @@ void task_button( void * pvParameters ){ | |||||||
| //============== fan task ===============
 | //============== fan task ===============
 | ||||||
| //=======================================
 | //=======================================
 | ||||||
| //task that controlls fans for cooling the drivers
 | //task that controlls fans for cooling the drivers
 | ||||||
| void task_fans( void * pvParameters ){ | //void task_fans( void * pvParameters ){
 | ||||||
| 	ESP_LOGI(TAG, "Initializing fans and starting fan handle loop"); | //	ESP_LOGI(TAG, "Initializing fans and starting fan handle loop");
 | ||||||
| 	//create fan instances with config defined in config.cpp
 | //	//create fan instances with config defined in config.cpp
 | ||||||
| 	controlledFan fan(configCooling, &motorLeft, &motorRight); | //	controlledFan fan(configCooling, &motorLeft, &motorRight);
 | ||||||
| 	//repeatedly run fan handle function in a slow loop
 | //	//repeatedly run fan handle function in a slow loop
 | ||||||
| 	while(1){ | //	while(1){
 | ||||||
| 		fan.handle(); | //		fan.handle();
 | ||||||
| 		vTaskDelay(500 / portTICK_PERIOD_MS); | //		vTaskDelay(500 / portTICK_PERIOD_MS);
 | ||||||
| 	} | //	}
 | ||||||
| } | //}
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -142,7 +142,10 @@ void setLoglevels(void){ | |||||||
| 	esp_log_level_set("wifi", ESP_LOG_INFO); | 	esp_log_level_set("wifi", ESP_LOG_INFO); | ||||||
| 	esp_log_level_set("http", ESP_LOG_INFO); | 	esp_log_level_set("http", ESP_LOG_INFO); | ||||||
| 	esp_log_level_set("automatedArmchair", ESP_LOG_DEBUG); | 	esp_log_level_set("automatedArmchair", ESP_LOG_DEBUG); | ||||||
| 	//esp_log_level_set("current-sensors", ESP_LOG_INFO);
 | 		esp_log_level_set("uart_common", ESP_LOG_INFO); | ||||||
|  | 	esp_log_level_set("uart", ESP_LOG_INFO); | ||||||
|  | 
 | ||||||
|  | 	//
 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| @ -166,25 +169,25 @@ extern "C" void app_main(void) { | |||||||
| 	//------------------------------
 | 	//------------------------------
 | ||||||
| 	//--- create task for buzzer ---
 | 	//--- create task for buzzer ---
 | ||||||
| 	//------------------------------
 | 	//------------------------------
 | ||||||
| 	xTaskCreate(&task_buzzer, "task_buzzer", 2048, NULL, 2, NULL); | 	//xTaskCreate(&task_buzzer, "task_buzzer", 2048, NULL, 2, NULL);
 | ||||||
| 
 | 
 | ||||||
| 	//-------------------------------
 | 	//-------------------------------
 | ||||||
| 	//--- create task for control ---
 | 	//--- create task for control ---
 | ||||||
| 	//-------------------------------
 | 	//-------------------------------
 | ||||||
| 	//task that generates motor commands depending on the current mode and sends those to motorctl task
 | 	//task that generates motor commands depending on the current mode and sends those to motorctl task
 | ||||||
| 	xTaskCreate(&task_control, "task_control", 4096, NULL, 5, NULL); | 	xTaskCreate(&task_control, "task_control", 5*4096, NULL, 5, NULL); | ||||||
| 
 | 
 | ||||||
| 	//------------------------------
 | 	//------------------------------
 | ||||||
| 	//--- create task for button ---
 | 	//--- create task for button ---
 | ||||||
| 	//------------------------------
 | 	//------------------------------
 | ||||||
| 	//task that evaluates and processes the button input and runs the configured commands
 | 	//task that evaluates and processes the button input and runs the configured commands
 | ||||||
| 	xTaskCreate(&task_button, "task_button", 4096, NULL, 4, NULL); | 	//xTaskCreate(&task_button, "task_button", 4096, NULL, 4, NULL);
 | ||||||
| 
 | 
 | ||||||
| 	//-----------------------------------
 | 	//-----------------------------------
 | ||||||
| 	//--- create task for fan control ---
 | 	//--- create task for fan control ---
 | ||||||
| 	//-----------------------------------
 | 	//-----------------------------------
 | ||||||
| 	//task that evaluates and processes the button input and runs the configured commands
 | 	//task that evaluates and processes the button input and runs the configured commands
 | ||||||
| 	xTaskCreate(&task_fans, "task_fans", 2048, NULL, 1, NULL); | 	//xTaskCreate(&task_fans, "task_fans", 2048, NULL, 1, NULL);
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	//beep at startup
 | 	//beep at startup
 | ||||||
| @ -213,16 +216,22 @@ extern "C" void app_main(void) { | |||||||
| 	//    http_init_server();
 | 	//    http_init_server();
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	//--- testing force http mode after startup ---
 |  | ||||||
| 	//control.changeMode(controlMode_t::HTTP);
 |  | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
|  | 	//-------------------------------------------
 | ||||||
|  | 	//--- create tasks for uart communication ---
 | ||||||
|  | 	//-------------------------------------------
 | ||||||
|  | 
 | ||||||
| 	uart_init(); | 	uart_init(); | ||||||
| 	xTaskCreate(task_uartReceive, "task_uartReceive", 4096, NULL, 10, NULL); | 	xTaskCreate(task_uartReceive, "task_uartReceive", 4096, NULL, 10, NULL); | ||||||
| 	xTaskCreate(task_uartSend, "task_uartSend", 4096, NULL, 10, NULL); | 	xTaskCreate(task_uartSend, "task_uartSend", 4096, NULL, 10, NULL); | ||||||
| 
 | 
 | ||||||
| 	//--- main loop ---
 | 	//--- main loop ---
 | ||||||
| 	//does nothing except for testing things
 | 	//does nothing except for testing things
 | ||||||
|  | 	 | ||||||
|  | 	//--- testing force http mode after startup ---
 | ||||||
|  | 		vTaskDelay(5000 / portTICK_PERIOD_MS); | ||||||
|  | 	control.changeMode(controlMode_t::HTTP); | ||||||
| 	while(1){ | 	while(1){ | ||||||
| 		vTaskDelay(1000 / portTICK_PERIOD_MS); | 		vTaskDelay(1000 / portTICK_PERIOD_MS); | ||||||
| 		//---------------------------------
 | 		//---------------------------------
 | ||||||
|  | |||||||
| @ -20,12 +20,19 @@ extern "C" | |||||||
| static const char * TAG = "uart"; | static const char * TAG = "uart"; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | //TESTING
 | ||||||
|  | #include "control.hpp" | ||||||
|  | #include "config.hpp" | ||||||
| 
 | 
 | ||||||
| //==============================
 | //==============================
 | ||||||
| //====== task_uartReceive ======
 | //====== task_uartReceive ======
 | ||||||
| //==============================
 | //==============================
 | ||||||
| //TODO copy receive task from board_motorctl/uart.cpp
 | //TODO copy receive task from board_motorctl/uart.cpp
 | ||||||
| void task_uartReceive(void *arg){ | void task_uartReceive(void *arg){ | ||||||
|  | 	//--- testing force http mode after startup ---
 | ||||||
|  | 	//TESTING
 | ||||||
|  | 	vTaskDelay(5000 / portTICK_PERIOD_MS); | ||||||
|  | 	control.changeMode(controlMode_t::HTTP); | ||||||
| 	while (1) { | 	while (1) { | ||||||
| 		vTaskDelay(200 / portTICK_PERIOD_MS); | 		vTaskDelay(200 / portTICK_PERIOD_MS); | ||||||
| 	} | 	} | ||||||
| @ -42,7 +49,7 @@ void task_uartSend(void *arg){ | |||||||
| 	uartData_test_t data = {123, 0, 1.1}; | 	uartData_test_t data = {123, 0, 1.1}; | ||||||
| 	ESP_LOGW(TAG, "startloop..."); | 	ESP_LOGW(TAG, "startloop..."); | ||||||
| 	while (1) { | 	while (1) { | ||||||
| 		vTaskDelay(500 / portTICK_PERIOD_MS);     | 		vTaskDelay(10000 / portTICK_PERIOD_MS);     | ||||||
| 		uart_sendStruct<uartData_test_t>(data); | 		uart_sendStruct<uartData_test_t>(data); | ||||||
| 
 | 
 | ||||||
| 		//change data values
 | 		//change data values
 | ||||||
|  | |||||||
| @ -103,6 +103,8 @@ void setLoglevels(void){ | |||||||
| 	esp_log_level_set("wifi", ESP_LOG_INFO); | 	esp_log_level_set("wifi", ESP_LOG_INFO); | ||||||
| 	esp_log_level_set("http", ESP_LOG_INFO); | 	esp_log_level_set("http", ESP_LOG_INFO); | ||||||
| 	esp_log_level_set("automatedArmchair", ESP_LOG_DEBUG); | 	esp_log_level_set("automatedArmchair", ESP_LOG_DEBUG); | ||||||
|  | 	esp_log_level_set("uart_common", ESP_LOG_INFO); | ||||||
|  | 	esp_log_level_set("uart", ESP_LOG_INFO); | ||||||
| 	//esp_log_level_set("current-sensors", ESP_LOG_INFO);
 | 	//esp_log_level_set("current-sensors", ESP_LOG_INFO);
 | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
| @ -146,11 +148,12 @@ extern "C" void app_main(void) { | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| #ifdef UART_TEST_ONLY | 	//-------------------------------------------
 | ||||||
|  | 	//--- create tasks for uart communication ---
 | ||||||
|  | 	//-------------------------------------------
 | ||||||
| 	uart_init(); | 	uart_init(); | ||||||
| 	xTaskCreate(task_uartReceive, "task_uartReceive", 4096, NULL, 10, NULL); | 	xTaskCreate(task_uartReceive, "task_uartReceive", 4096, NULL, 10, NULL); | ||||||
| 	xTaskCreate(task_uartSend, "task_uartSend", 4096, NULL, 10, NULL); | 	xTaskCreate(task_uartSend, "task_uartSend", 4096, NULL, 10, NULL); | ||||||
| #endif |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 	//--- main loop ---
 | 	//--- main loop ---
 | ||||||
|  | |||||||
| @ -1,4 +1,6 @@ | |||||||
| #include "motorctl.hpp" | #include "motorctl.hpp" | ||||||
|  | #include "esp_log.h" | ||||||
|  | #include "types.hpp" | ||||||
| 
 | 
 | ||||||
| //tag for logging
 | //tag for logging
 | ||||||
| static const char * TAG = "motor-control"; | static const char * TAG = "motor-control"; | ||||||
| @ -80,6 +82,8 @@ void controlledMotor::handle(){ | |||||||
|         ESP_LOGD(TAG, "Read command from queue: state=%s, duty=%.2f", motorstateStr[(int)commandReceive.state], commandReceive.duty); |         ESP_LOGD(TAG, "Read command from queue: state=%s, duty=%.2f", motorstateStr[(int)commandReceive.state], commandReceive.duty); | ||||||
|         state = commandReceive.state; |         state = commandReceive.state; | ||||||
|         dutyTarget = commandReceive.duty; |         dutyTarget = commandReceive.duty; | ||||||
|  | 		receiveTimeout = false; | ||||||
|  | 		timestamp_commandReceived = esp_log_timestamp(); | ||||||
| 
 | 
 | ||||||
|         //--- convert duty ---
 |         //--- convert duty ---
 | ||||||
|         //define target duty (-100 to 100) from provided duty and motorstate
 |         //define target duty (-100 to 100) from provided duty and motorstate
 | ||||||
| @ -102,6 +106,14 @@ void controlledMotor::handle(){ | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | 	//--- timeout, no data ---
 | ||||||
|  | 	//turn motors off if no data received for long time (e.g. no uart data / control offline)
 | ||||||
|  | 	if ((esp_log_timestamp() - timestamp_commandReceived) > 3000 && !receiveTimeout){ | ||||||
|  | 		receiveTimeout = true; | ||||||
|  | 		state = motorstate_t::IDLE; | ||||||
|  | 		dutyTarget = 0; | ||||||
|  | 		ESP_LOGE(TAG, "TIMEOUT, no target data received for more than 3s -> switch to IDLE"); | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
|     //--- calculate increment ---
 |     //--- calculate increment ---
 | ||||||
|     //calculate increment for fading UP with passed time since last run and configured fade time
 |     //calculate increment for fading UP with passed time since last run and configured fade time
 | ||||||
|  | |||||||
| @ -77,4 +77,7 @@ class controlledMotor { | |||||||
| 
 | 
 | ||||||
|         struct motorCommand_t commandReceive = {}; |         struct motorCommand_t commandReceive = {}; | ||||||
|         struct motorCommand_t commandSend = {}; |         struct motorCommand_t commandSend = {}; | ||||||
|  | 
 | ||||||
|  | 		uint32_t timestamp_commandReceived = 0; | ||||||
|  | 		bool receiveTimeout = false; | ||||||
| }; | }; | ||||||
|  | |||||||
| @ -1,4 +1,6 @@ | |||||||
| #include "uart.hpp" | #include "uart.hpp" | ||||||
|  | #include "config.hpp" | ||||||
|  | #include "types.hpp" | ||||||
| //===== uart board MOTORCTL =====
 | //===== uart board MOTORCTL =====
 | ||||||
| 
 | 
 | ||||||
| static const char * TAG = "uart"; | static const char * TAG = "uart"; | ||||||
| @ -8,24 +10,39 @@ static const char * TAG = "uart"; | |||||||
| //====== task_uartReceive ======
 | //====== task_uartReceive ======
 | ||||||
| //==============================
 | //==============================
 | ||||||
| void task_uartReceive(void *arg){ | void task_uartReceive(void *arg){ | ||||||
|  | 	ESP_LOGW(TAG, "receive task started"); | ||||||
| 	//receive data from uart, detect associated struct and copy/handle the data
 | 	//receive data from uart, detect associated struct and copy/handle the data
 | ||||||
| 	//TODO use queue instead of check interval?
 | 	//TODO use queue instead of check interval?
 | ||||||
| 	uartData_test_t testData; | 	uartData_test_t dataTest; | ||||||
|  | 	motorCommands_t dataMotorCommands; | ||||||
| 	uint8_t receivedData[1024-1]; | 	uint8_t receivedData[1024-1]; | ||||||
| 	while(1){ | 	while(1){ | ||||||
| 		//note: check has to be more frequent than pause time between sending
 | 		//note: check has to be more frequent than pause time between sending
 | ||||||
| 		vTaskDelay(200 / portTICK_PERIOD_MS); | 		vTaskDelay(50 / portTICK_PERIOD_MS); | ||||||
| 		int len = uart_read_bytes(UART_NUM_1, receivedData, sizeof(uartData_test_t), 20 / portTICK_PERIOD_MS); | 		//read bytes (max 1023) until 20ms pause is happening
 | ||||||
|  | 		int len = uart_read_bytes(UART_NUM_1, receivedData, sizeof(receivedData), 20 / portTICK_PERIOD_MS); | ||||||
| 		uart_flush_input(UART_NUM_1); | 		uart_flush_input(UART_NUM_1); | ||||||
| 		if (len < 1) continue; | 		if (len < 1) continue; | ||||||
| 		switch (len){ | 		switch (len){ | ||||||
|  | 
 | ||||||
| 			case sizeof(uartData_test_t): | 			case sizeof(uartData_test_t): | ||||||
| 				testData = serialData2Struct<uartData_test_t>(receivedData); | 				dataTest = serialData2Struct<uartData_test_t>(receivedData); | ||||||
| 				ESP_LOGW(TAG, "received uartDataStruct len=%d DATA: timestamp=%d, id=%d, value=%.1f", len, testData.timestamp, testData.id, testData.value); | 				ESP_LOGW(TAG, "received uartDataStruct len=%d DATA: timestamp=%d, id=%d, value=%.1f", len, dataTest.timestamp, dataTest.id, dataTest.value); | ||||||
| 				break; | 				break; | ||||||
|  | 
 | ||||||
|  | 			case sizeof(motorCommands_t): | ||||||
|  | 				dataMotorCommands = serialData2Struct<motorCommands_t>(receivedData); | ||||||
|  | 				ESP_LOGI(TAG, "received motorCommands struct len=%d left=%.2f%% right=%.2f%%, update target...", len, dataMotorCommands.left.duty, dataMotorCommands.right.duty); | ||||||
|  | 				//update target motor state and duty
 | ||||||
|  | 				motorLeft.setTarget(dataMotorCommands.left.state,  | ||||||
|  | 						dataMotorCommands.left.duty); | ||||||
|  | 				motorRight.setTarget(dataMotorCommands.right.state,  | ||||||
|  | 						dataMotorCommands.right.duty); | ||||||
|  | 				break; | ||||||
|  | 
 | ||||||
| 				//TODO add other received structs here
 | 				//TODO add other received structs here
 | ||||||
| 			default: | 			default: | ||||||
| 				ESP_LOGW(TAG, "received data len=%d cant be associated with configures struct", len); | 				ESP_LOGE(TAG, "received data len=%d cant be associated with configures struct", len); | ||||||
| 				break; | 				break; | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -38,6 +55,7 @@ void task_uartReceive(void *arg){ | |||||||
| //=============================
 | //=============================
 | ||||||
| //TODO copy send task from board_control/uart.cpp
 | //TODO copy send task from board_control/uart.cpp
 | ||||||
| void task_uartSend(void *arg){ | void task_uartSend(void *arg){ | ||||||
|  | 	ESP_LOGW(TAG, "send task started"); | ||||||
| 	while (1) { | 	while (1) { | ||||||
| 		vTaskDelay(500 / portTICK_PERIOD_MS);     | 		vTaskDelay(500 / portTICK_PERIOD_MS);     | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -3,6 +3,7 @@ idf_component_register( | |||||||
|         "wifi.c" |         "wifi.c" | ||||||
| 		"buzzer.cpp" | 		"buzzer.cpp" | ||||||
| 		"uart_common.cpp" | 		"uart_common.cpp" | ||||||
|  | 		"types.cpp" | ||||||
|     INCLUDE_DIRS  |     INCLUDE_DIRS  | ||||||
|         "." |         "." | ||||||
| 		PRIV_REQUIRES nvs_flash | 		PRIV_REQUIRES nvs_flash | ||||||
|  | |||||||
| @ -19,7 +19,7 @@ extern "C" | |||||||
| //===============================
 | //===============================
 | ||||||
| enum class motorstate_t {IDLE, FWD, REV, BRAKE}; | enum class motorstate_t {IDLE, FWD, REV, BRAKE}; | ||||||
| //definition of string array to be able to convert state enum to readable string (defined in motordrivers.cpp)
 | //definition of string array to be able to convert state enum to readable string (defined in motordrivers.cpp)
 | ||||||
| const char* motorstateStr[] = {"IDLE", "FWD", "REV", "BRAKE"}; | extern const char* motorstateStr[4]; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -15,8 +15,9 @@ extern "C" | |||||||
| #include "freertos/queue.h" | #include "freertos/queue.h" | ||||||
| #include "driver/uart.h" | #include "driver/uart.h" | ||||||
| } | } | ||||||
|  | #include "types.hpp" | ||||||
| 
 | 
 | ||||||
| //struct for testin uart
 | //struct for testing uart
 | ||||||
| typedef struct { | typedef struct { | ||||||
| 	uint32_t timestamp; | 	uint32_t timestamp; | ||||||
| 	int id; | 	int id; | ||||||
| @ -24,6 +25,13 @@ typedef struct { | |||||||
| } uartData_test_t; | } uartData_test_t; | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | //unnecessary, using commands struct directly
 | ||||||
|  | typedef struct { | ||||||
|  | 	uint32_t timestamp; | ||||||
|  | 	motorCommands_t commands; | ||||||
|  | } uartData_motorCommands_t; | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| //===== uart_init =====
 | //===== uart_init =====
 | ||||||
| //should be run once at startup
 | //should be run once at startup
 | ||||||
| void uart_init(void); | void uart_init(void); | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user