日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 综合教程 >内容正文

综合教程

[如何构建自己的轮式移动机器人系统·从入门到放弃]机器人底层篇

發布時間:2024/8/5 综合教程 48 生活家
生活随笔 收集整理的這篇文章主要介紹了 [如何构建自己的轮式移动机器人系统·从入门到放弃]机器人底层篇 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

大家好,在下又回來了。今天開一個新坑,算是小小地總結一下之前的工作。

在這個系列教程中,我會嘗試教大家一步一步從底層開始,構建屬于自己的移動機器人。為了開發的簡單方便,上層使用了裝有ROS(robot operatingsystem)的linux板卡和臺式電腦(臺式機),而下層使用了STM32F407作為MCU的嵌入式系統,并且使用了開源項目HANDS-FREE[2]的大部分代碼并在此基礎上做了一些移植工作。

直入主題。輪式機器人可以簡單定義為以輪子作為運動機構的移動機器人??梢詫⑤喪綑C器人分為兩輪(平衡車),三輪和四輪,當然還有六輪的形式。其實都大同小異。

下表引自[1]。

除了兩輪車外,其他輪式的移動機器人都是天然魯棒系統,所以這對高性能控制的要求幾乎為零。

明顯地,這是一個項目教程,因此我將以三輪全向輪式移動機器人為例,講一講如何從零完成一個機器人項目。

第一部分:機械

淘寶搜索“全向移動機器人”,本例中用的是這款

注意,電機為普通的直流電機,因為是室內移動機器人,機器人的速度小于1m/s,速度不需要很高。要求每個電機上都要有里程計(編碼器),目的是獲得小車上每個電機的里程與比較粗糙的速度信息。

第二部分:嵌入式系統-硬件

為了解除廣告之嫌,我不會放鏈接。如果有興趣請留言聯系我。

首先,為了要控制電機,需要兩到三個電機驅動。我選擇的是具有L298N邏輯的雙路電機驅動,滿足三個電機的控制要求。淘寶上隨便找種L298N模塊即可。

電源系統:需要給單片機最小系統板提供5v供電,僅需要一個能提供5v的LM25XX開關電源模塊即可。非常輕松愉快。

單片機最小系統板:一個STM32F407VET6最小系統板即可。需要板上自帶能提供3.3v的LDO,還有三到四個開關和兩盞LED,可以不需要RTC電池。

在Embedded/doc/下會此版的原理圖。

傳感器:最少需要一個MPU6050模塊,一個HMC5883L模塊。這是為了融合出機器人的yaw,這是三個機器人的狀態變量之一。單獨一個MPU6050測出的yaw角度會隨著時間漂移,需要HMC5883磁力計矯正。最好可以有一個到兩個普通的超聲波測距模塊(如HC-SR04),主要作為壁障用。下圖是集成了MPU6050,HMC5883,BMP180的GY-87模塊

電池:12V動力鋰電池即可。

其他:①舵機:普通數字舵機或者高級的帶串口的數字舵機均可,可以組成簡單機械臂或者機器人頭;

第三部分:嵌入式系統-軟件

本例中的代碼結構和大部分的源代碼來自開源工程HANDS-FREE。它是一個由西北工大的團隊維護的開源項目。在本文中,我會比較詳盡的對代碼進行講解。

首先整體介紹以下嵌入式軟件部分。這里沒有使用實時操作系統。開發環境為keil-ARM V5.17+ windows 10 enterprise。使用C++編程。

[這里為嵌入式軟件整體功能以及算法流程圖]

工程文件結構如下

USER文件夾內主要是用戶文件。Start_Code文件夾內主要是底層庫函數代碼。而BSP文件夾內放有用戶庫函數。剩下的就是外設功能包。

首先介紹一下主函數。

//main.cpp
int main(void) { constructorInit(); systemInit(); while(1) { if(usart1_queue.emptyCheck()==0){ hf_link_pc_node.byteAnalysisCall(usart1_queue.getData()); } if ( board.cnt_1ms >= 1 ) // 1000hz { board.cnt_1ms=0; imu.topCall(); // stm32f4 -- 631us(fpu) } if ( board.cnt_2ms >= 2 ) // 500hz { board.cnt_2ms=0; } if ( board.cnt_5ms >= 5 ) // 200hz { board.cnt_5ms=0; } if ( board.cnt_10ms >= 10 ) // 100hz { board.cnt_10ms=0; board.boardBasicCall(); // need time stm32f1 35us } if ( board.cnt_20ms >= 20 ) // 50hz { board.cnt_20ms = 0 ; motor_top.motorTopCall(); //motor speed control } if ( board.cnt_50ms >= 50 ) // 20hz { board.cnt_50ms = 0 ; //robot_head.headTopCall(); hands_free_robot.robotWheelTopCall(); //robot control interface } if ( board.cnt_500ms >= 500 ) // 2hz { board.cnt_500ms = 0 ; //robot_head.headTopCall(); board.setLedState(0,2); } } return 0; }

可以大致看到整個機器人底層的執行流程:先進行系統架構的初始化和系統外設初始化。然后進入主循環:每一毫秒都會檢查接收串口1數據的隊列是否有數,如果有數據就對數據幀作分析,否則就更新imu的數據并處理(或者只將數據更新之后交由上位機處理);每10ms更新一次系統數據(CPU溫度,系統時間和電池電壓);每20ms控制一次電機(對電機底層的控制);每50ms控制一次機器人運動(對機器人姿態的控制);每500ms翻轉一下led電平(指示系統正在運行)。

下面簡單講解一下系統架構初始化和系統外設初始化函數。

下面是系統外設初始化。

//main.cpp
void constructorInit(void) { board = Board(); Ultrasonic = ULTRASONIC(); my_robot = RobotAbstract(); motor_top = MotorTop(); //robot_head = HeadAX(); hands_free_robot = RobotWheel(); //sbus = SBUS(); imu = IMU(); usart1_queue = Queue(); hf_link_pc_node = HFLink(0x11,0x01,&my_robot); hf_link_node_pointer=&hf_link_pc_node; }

這里將描述系統外設的所有抽象類進行初始化,我將以board類為例大致講一講C++在嵌入式編程中的具體運用,因為確實好用。

//board.h
class Board { public: float cpu_temperature; float cpu_usage; float battery_voltage; float system_time; //system working time (unit:us), start after power-on uint8_t system_init; //state of system: 0-->not initialize 1-->initialized uint16_t cnt_1ms; uint16_t cnt_2ms; uint16_t cnt_5ms; uint16_t cnt_10ms; uint16_t cnt_20ms; uint16_t cnt_50ms; uint16_t cnt_500ms; uint32_t chipUniqueID[3]; uint16_t flashSize; //Unit: KB public: Board(); void boardBasicInit(void); void boardBasicCall(void); /**********************************************************************************************************************/ void debugPutChar(uint8_t tx_byte_); //system support functions void setLedState(uint8_t led_id, uint8_t operation); void setBeepState(uint8_t operation); uint8_t getKeyState(uint8_t key_id){return key_state[key_id] ;} /**********************************************************************************************************************/ void motorInterfaceInit(uint8_t motor_id , float motor_pwm_T); //package "PAKG_MOTOR" support functions void motorEnable(uint8_t motor_id); void motorDisable(uint8_t motor_id); void motorEnableForward(uint8_t motor_id); void motorEnableBackward(uint8_t motor_id); float getMotorEncoderCNT(uint8_t motor_id); float getMotorCurrent(uint8_t motor_id); void motorSetPWM(uint8_t motor_id , int pwm_value); /**********************************************************************************************************************/ void axServoInterfaceInit(void); //package " PAKG_SERVO" support functions void axServoTxModel(void); void axServoRxModel(void); void axServoSendTxByte(uint8_t tx_byte); uint8_t axServoGetRxByte(uint8_t *error); /**********************************************************************************************************************/ void sbusInterfaceInit(void); //package " SBUS_PPM" support functions void ppmInterfaceInit(void); /**********************************************************************************************************************/ //package " ANALOG_SERVO" support functions void analogServoInterfaceInit(void); void analogServoSetPWM(uint8_t servo_id , int pwm_value); /**********************************************************************************************************************/ void imuI2CInit(void); //package "PAKG_IMU" support functions void imuI2CWriteByte(uint8_t equipment_address , uint8_t reg_address , uint8_t reg_data , uint8_t fastmode); uint8_t imuI2CReadByte(uint8_t equipment_address , uint8_t reg_address , uint8_t fastmode); uint8_t imuI2CWriteBuf(uint8_t equipment_address,uint8_t reg_address, uint8_t* pt_char , uint8_t size , uint8_t fastmode); uint8_t imuI2CReadBuf(uint8_t equipment_address,uint8_t reg_address, uint8_t * pt_char , uint8_t size , uint8_t fastmode); void gpsInterfaceInit(void); void gpsSendTxByte(uint8_t tx_byte); /**********************************************************************************************************************/ void eepromI2CInit(void); //package "AT24Cxx" support functions void eepromI2CWriteByte(uint8_t equipment_address , uint8_t reg_address , uint8_t reg_data , uint8_t fastmode); uint8_t eepromI2CReadByte(uint8_t equipment_address , uint8_t reg_address , uint8_t fastmode); uint8_t eepromI2CWriteBuf(uint8_t equipment_address,uint8_t reg_address, uint8_t* pt_char , uint8_t size , uint8_t fastmode); uint8_t eepromI2CReadBuf(uint8_t equipment_address,uint8_t reg_address, uint8_t * pt_char , uint8_t size , uint8_t fastmode); /**********************************************************************************************************************/ //extend interface support functions void extendI2CInit(void); // extend iic interface is using for extend other sensors void extendI2CWriteByte(uint8_t equipment_address , uint8_t reg_address , uint8_t reg_data , uint8_t fastmode); uint8_t extendI2CReadByte(uint8_t equipment_address , uint8_t reg_address , uint8_t fastmode); uint8_t extendI2CWriteBuf(uint8_t equipment_address,uint8_t reg_address, uint8_t* pt_char , uint8_t size , uint8_t fastmode); uint8_t extendI2CReadBuf(uint8_t equipment_address,uint8_t reg_address, uint8_t* pt_char , uint8_t size , uint8_t fastmode); //extend spi interface //extend can interface //extend PWM interface //mode=1 high frequency control ; mode = 2 for electric modulation void pwmInterfaceInit(uint8_t mode , float pwm_t); void setPWMValue(uint8_t channel_x , float pwm_value); //channel_x 1~6 /**********************************************************************************************************************/\ private: float battery_voltage_; float battery_voltage_alarm_ ; float battery_proportion_ ; float temperature_; uint16_t beep_alarm_cnt_ ; uint16_t board_call_i , board_call_j , board_call_k; uint8_t key_state[5]; void systemMonitoring(void); void ledInit(void); void keyInit(void); void keyStateRenew(void); void beepInit(void); void debugInterfaceInit(void); float getBatteryVoltage(void); float getCPUUsage(void); float getCPUTemperature(void); void getCPUInfo(void); }; extern Board board;

C++是一種面向對象的編程語言,C++ 在 C 語言的基礎上增加了面向對象編程,C++ 支持面向對象程序設計。類是 C++ 的核心特性,通常被稱為用戶定義的類型。[3]

類本身是抽象的。定義一個類本質上是定義了一個數據類型的藍圖。程序員將一種事物定義為一種類,然后把這個類的屬性與能進行的操作包含在類下。如此例,定義board為描述主控板的類。public標識符下的數據成員為共有的,也就是說其他的類可以調用這些數據成員。而private下則是私有的,也就是說這些數據成員只能在這個類里面使用。主要的原因是C++數據封裝與信息隱藏的功能。在大型項目的開發中,代碼應該簡明并且合理,應該在類里完成的工作就讓他在類里面完成就行了。

在這個類下,定義了板子的初始化函數(這里初始化了單片機的時鐘和開發板的外設,如定時器、key、led、蜂鳴器、ADC模塊等)

//control_unit_v2.cpp
void Board::boardBasicInit(void) { int i; for(i=0;i<0x8fff;i++); HF_System_Timer_Init(); //Initialize the measurement systemm debugInterfaceInit(); ledInit(); keyInit(); beepInit(); //HF_RTC_Init(); //Initialize the RTC, if return 0:failed,else if return 1:succeeded //HF_IWDG_Init(); //Initialize the independed watch dog, system will reset if not feeding dog over 1s HF_ADC_Moder_Init(0X3E00 , 5 , 2.5f); //ADC init HF_Timer_Init(TIM6 , 1000); //timer6 init , 1000us setBeepState(1); delay_ms(500); setBeepState(0); analogServoInterfaceInit(); }

后面還提供了可供用戶調用的其他外設包的函數,比如設定led狀態的

    void setLedState(uint8_t led_id, uint8_t operation);

和設定使能和失能電機的

    void motorEnable(uint8_t motor_id);
    void motorDisable(uint8_t motor_id);

其他類的具體實現可以參考其源碼。

回到主函數,我們這時其實還沒有進行任何初始化,只是對類進行了個“抽象”的初始化(具體的初始化還沒有實現),接下來就開始進行真正的硬件外設初始化。

//main.cpp
void systemInit(void) { //INTX_DISABLE(); //close all interruption board.boardBasicInit(); motor_top.motorTopInit(3 , 1560 , 0.02 , 0); //robot_head.axServoInit(); hands_free_robot.robotWheelTopInit(); //sbus.sbusInit(); imu.topInit(1,1,1,0,0,1); Ultrasonic.Ultra_Init(1); //INTX_ENABLE(); //enable all interruption printf("app start \r\n"); }

這里的所有函數除了printf,都是各種外設的初始化。上面已經提到了boardBasicInit(),那就以Motor_top類下的motorTopInit()為例吧。

//motor_top.cpp
void MotorTop::motorTopInit(float motor_enable_num_ , float motor_encoder_num_ , float motor_pid_t_ , unsigned char motor_simulation_model_) { motor_enable_num = motor_enable_num_; motor_encoder_num = motor_encoder_num_; motor_pid_t = motor_pid_t_; board.motorInterfaceInit(1, motor_pwm_max); //motor_x init motor1.motorControlInit(motor_pid_t_ , motor_encoder_num_ , motor_pwm_max , motor_simulation_model_); motorPWMRenew(1,0); board.motorInterfaceInit(2, motor_pwm_max); motor2.motorControlInit(motor_pid_t_ , motor_encoder_num_ , motor_pwm_max , motor_simulation_model_); motorPWMRenew(2,0); if(motor_enable_num >= 3) { board.motorInterfaceInit(3, motor_pwm_max); motor3.motorControlInit(motor_pid_t_ , motor_encoder_num_ , motor_pwm_max , motor_simulation_model_); motorPWMRenew(3,0); } if(motor_enable_num >= 4) { board.motorInterfaceInit(4, motor_pwm_max); motor4.motorControlInit(motor_pid_t_ , motor_encoder_num_ , motor_pwm_max , motor_simulation_model_); motorPWMRenew(4,0); } }

這個函數有兩大功能,一是初始化STM32的TIM1,二是初始化里程計(編碼器)用到的TIM2、TIM3、TIM4定時器。具體的實現函數為motorInterfaceInit()

//control_unit_v2.cpp
void Board::motorInterfaceInit(uint8_t motor_id , float motor_pwm_T) { GPIO_InitTypeDef GPIO_InitStructure; GPIO_StructInit(&GPIO_InitStructure); GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_DOWN; if(motor_id == 1 ){ RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOE , ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_10; GPIO_Init(GPIOE , &GPIO_InitStructure); GPIO_ResetBits(GPIOE , GPIO_Pin_8); GPIO_ResetBits(GPIOE , GPIO_Pin_10); HF_Encoder_Init(TIM2,0); //encoder init for PID speed control } else if(motor_id == 2){ RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOC , ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1; GPIO_Init(GPIOC , &GPIO_InitStructure); GPIO_ResetBits(GPIOC , GPIO_Pin_0); GPIO_ResetBits(GPIOC , GPIO_Pin_1); HF_Encoder_Init(TIM3,2); } else if(motor_id == 3){ RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOC , ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3; GPIO_Init(GPIOC , &GPIO_InitStructure); GPIO_ResetBits(GPIOC , GPIO_Pin_2); GPIO_ResetBits(GPIOC , GPIO_Pin_3); HF_Encoder_Init(TIM4,1); } else if(motor_id == 4){ RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOE , ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15; GPIO_Init(GPIOE , &GPIO_InitStructure); GPIO_ResetBits(GPIOE , GPIO_Pin_15); HF_Encoder_Init(TIM5,0); } //motor_pwm_T = 5000 , TIM1 motor pwm frequency = (168M/4) / motor_pwm_T = 16.8K HF_PWMOut_Init(TIM1 , 2-1 , motor_pwm_T , 1); // //motor_pwm_T = 5000 , TIM9 motor pwm frequency = (168M/4) / motor_pwm_T = 16.8K // HF_PWMOut_Init(TIM9 , 2-1 , motor_pwm_T , 0); // //motor_pwm_T = 5000 , TIM12 motor pwm frequency = (84M/2) / motor_pwm_T = 16.8K // HF_PWMOut_Init(TIM12 , 0 , motor_pwm_T , 0); }

因為使用了L298N邏輯的電機驅動,所以一路電機需要兩個GPIO和一路PWM控制,因此上面這個函數主要是初始化使用的gpio和pwm功能的定時器TIM1,因為TIM1能發出四路不同的PWM信號,因此,三個電機僅僅需要一個TIM1。編碼器也用到了TIM,具體的代碼解析就不做了,[4]為介紹STM32編碼器接口庫函數的解析文檔,講得不錯。

回到systemInit函數。剩下分別初始化了motorTop的一些pid參數。這個函數屬于RobotWheel類,這個類主要包含了一些控制機器人姿態和運動的接口函數。它主要將上層發出的期望機器人速度或位置信號轉換成三個電機的控制信號。imu.topInit()主要完成了IMU的初始化。Ultra_Init完成了超聲波模塊的初始化。

接下來開始程序主循環的介紹。

首先每次都檢查隊列(queue)中的數據,調用queue.emptyCheck()。因為每次串口收到數據都會進入串口中斷接收數據并保存在隊列中,具體的代碼實現如下:

//stm32f4xx_it.cpp
void USART1_IRQHandler(void) { unsigned char data; if(USART1->SR&(1<<5)) { data=USART1->DR; if(usart1_queue.fullCheck()==0){ usart1_queue.putData(data); } USART_ClearITPendingBit(USART1, USART_IT_RXNE); // clear interrupt flag }

如果檢測到隊列中有數,就要開始對數據幀做解析了,實現函數如下。這個函數先將協議去除,獲得上位機發出的命令所代表的字,然后在通過調用packageAnalysis()對命令一一對應地做出相應。

//hf_link.cpp
unsigned char HFLink::byteAnalysisCall(const unsigned char rx_byte) { unsigned char package_update=0; unsigned char receive_message_update=0; receive_message_update=receiveFiniteStates(rx_byte) ; //Jump communication status if( receive_message_update ==1 ) { receive_message_update = 0; receive_message_count++; package_update=packageAnalysis(); if(package_update==1) receive_package_count++; return package_update; } return 0; }

下面就是去除協議幀的函數。它是通過一個狀態機來解析協議的。

//hf_link.cpp
unsigned char HFLink::receiveFiniteStates(const unsigned char rx_data) { switch (receive_state_) //協議解析狀態機 { case WAITING_FF1: if (rx_data == 0xff) //接收到幀頭第一個數據 { receive_state_ = WAITING_FF2; receive_check_sum_ =0; receive_message_length_ = 0; byte_count_=0; receive_check_sum_ += rx_data; } break;          //狀態機復位 case WAITING_FF2: if (rx_data == 0xff)    //接收到幀頭第二個數據,下面以此類推 { receive_state_ = SENDER_ID; receive_check_sum_ += rx_data; } else receive_state_ = WAITING_FF1; break; case SENDER_ID: rx_message_.sender_id = rx_data ; if (rx_message_.sender_id == friend_id) //id check { receive_check_sum_ += rx_data; receive_state_ = RECEIVER_ID; } else { receive_state_ = WAITING_FF1; } break; case RECEIVER_ID: rx_message_.receiver_id = rx_data ; if (rx_message_.receiver_id == my_id) //id check { receive_check_sum_ += rx_data; receive_state_ = RECEIVE_LEN_H; } else { receive_state_ = WAITING_FF1; } break; case RECEIVE_LEN_H: receive_check_sum_ += rx_data; receive_message_length_ |= rx_data<<8; receive_state_ = RECEIVE_LEN_L; break; case RECEIVE_LEN_L: receive_check_sum_ += rx_data; receive_message_length_ |= rx_data; rx_message_.length = receive_message_length_; receive_state_ = RECEIVE_PACKAGE; break; case RECEIVE_PACKAGE: receive_check_sum_ += rx_data; rx_message_.data[byte_count_++] = rx_data; if(byte_count_ >= receive_message_length_) { receive_state_ = RECEIVE_CHECK; receive_check_sum_=receive_check_sum_%255; } break; case RECEIVE_CHECK: if(rx_data == (unsigned char)receive_check_sum_) //對比發送和接收的校驗和,如果不一致直接放棄此數據幀 { receive_check_sum_=0; receive_state_ = WAITING_FF1; return 1 ;                     //傳回1,表示一個數據包收到 } else { receive_state_ = WAITING_FF1; } break; default: receive_state_ = WAITING_FF1; } return 0; }

這里的協議為:0XFF 0XFF sender_id receiver_id length_H length_L ****(data) check_sum。

解析完協議就要開始分析命令了。由packageAnalysis()來實現這一步工作。

//hf_link.cpp
unsigned char HFLink::packageAnalysis(void)
{

    unsigned char analysis_state=0;
    void* single_command;

    command_state_ = (Command)rx_message_.data[0];

    if (hf_link_node_model== 0)  //the slave need to check the SHAKING_HANDS"s state
    {
        if(shaking_hands_state==0 && command_state_ != SHAKING_HANDS) //if not  shaking hands
        {
            sendStruct(SHAKING_HANDS  , (unsigned char *)single_command, 0);
            return 1;
        }
    }

    switch (command_state_)
    {
    case SHAKING_HANDS :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_global_coordinate , sizeof(my_robot->measure_global_coordinate));
        break;

    case READ_ROBOT_SYSTEM_INFO :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->robot_system_info , sizeof(my_robot->robot_system_info));
        break;

    case SET_GLOBAL_SPEED :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->expect_global_speed , sizeof(my_robot->expect_global_speed));
        break;

    case READ_GLOBAL_SPEED :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_global_speed , sizeof(my_robot->measure_global_speed));
        break;

    case SET_ROBOT_SPEED :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->expect_robot_speed , sizeof(my_robot->expect_robot_speed));
        break;

    case READ_ROBOT_SPEED :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_robot_speed , sizeof(my_robot->measure_robot_speed));
        break;

    case SET_MOTOR_SPEED :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->expect_motor_speed, sizeof(my_robot->expect_motor_speed));
        break;

    case READ_MOTOR_SPEED :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_motor_speed , sizeof(my_robot->measure_motor_speed));
        break;

    case READ_MOTOR_MILEAGE :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_motor_mileage , sizeof(my_robot->measure_motor_mileage));
        break;

    case READ_GLOBAL_COORDINATE :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_global_coordinate , sizeof(my_robot->measure_global_coordinate));
        break;

    case READ_ROBOT_COORDINATE :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_robot_coordinate , sizeof(my_robot->measure_robot_coordinate));
        break;

    case CLEAR_COORDINATE_DATA :
        if (hf_link_node_model==0)
        {
            sendStruct(command_state_ , (unsigned char *)single_command , 0);
            analysis_state=1;
            receive_package_renew[(unsigned char)command_state_] = 1 ;
        }
        break;

    case SET_ARM_1 :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->expect_arm1_state, sizeof(my_robot->expect_arm1_state));
        break;

    case READ_ARM_1 :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_arm1_state , sizeof(my_robot->measure_arm1_state));
        break;

    case SET_ARM_2 :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->expect_arm2_state, sizeof(my_robot->expect_arm2_state));
        break;

    case READ_ARM_2 :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_arm2_state , sizeof(my_robot->measure_arm2_state));
        break;

    case SET_HEAD_1 :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->expect_head1_state , sizeof(my_robot->expect_head1_state ));
        break;

    case READ_HEAD_1 :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_head1_state , sizeof(my_robot->measure_head1_state));
        break;

    case SET_HEAD_2 :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->expect_head2_state, sizeof(my_robot->expect_head2_state));
        break;

    case READ_HEAD_2 :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_head2_state , sizeof(my_robot->measure_head2_state));
        break;

    case READ_IMU_DATA :
        analysis_state=readCommandAnalysis(command_state_ , (unsigned char *)&my_robot->measure_imu_euler_angle , sizeof(my_robot->measure_imu_euler_angle));
        break;

    case SET_ROBOY_PARAMETERS :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->robot_parameters, sizeof(my_robot->robot_parameters));
        break;

    case SAVE_ROBOY_PARAMETERS :
        if (hf_link_node_model==0)
        {
            sendStruct(command_state_ , (unsigned char *)single_command , 0);
            analysis_state=1;
            receive_package_renew[(unsigned char)command_state_] = 1 ;
        }
        break;

    case SET_MOTOR_PARAMETERS :
        analysis_state=setCommandAnalysis(command_state_ , (unsigned char *)&my_robot->motor_parameters, sizeof(my_robot->motor_parameters));
        break;

    case SAVE_MOTOR_PARAMETERS :
        if (hf_link_node_model==0)
        {
            sendStruct(command_state_ , (unsigned char *)single_command , 0);
            analysis_state=1;
            receive_package_renew[(unsigned char)command_state_] = 1 ;
        }
        break;

    default :
        analysis_state=0;
        break;

    }

    rx_message_.sender_id=0;    //clear flag
    rx_message_.receiver_id=0;
    rx_message_.length=0;
    rx_message_.data[0]=0;

    return analysis_state;
}

上位機在一開始時通常會發一幀握手信號,然后開始發送數據。然后在switch結構里,對應每一個命令都有相應的代碼做處理。大致可以將處理分為設定數據(setCommandAnalysis)和上傳數據(readCommandAnalysis)兩個函數。

結束后返回分析狀態,標志命令更新。在上述兩個函數里,還有對應的發送數據的函數,詳細請參考源碼。

將收到的數據獲得后,主循環進入了實際的控制模式。

首先最需要處理是IMU數據。下面的代碼將陀螺儀數據的傳輸頻率設定為500hz,bmp085為100hz。只是將數據傳輸給上位機,并沒有做融合處理。

//imu_top.cpp
void IMU::topCall(void) { imu_call_1++; imu_call_2++; imu_call_3++; imu_call_4++; imu_call_5++; if( imu_call_1 >= 2 ) //500HZ { imu_call_1=0; mpu6050.dataUpdate(); } if( imu_call_2 >= 5 ) //200HZ { imu_call_2=0; } if( imu_call_3 >= 10 ) //100HZ { imu_call_3 = 0; if(bmp085_en == 1) bmp085.dataUpdate(); if(ms611_en == 1) ms611.dataUpdate(); } if( imu_call_4 >= 20 ) //50HZ { imu_call_4=0; if(hmc085_en == 1) hmc5883l.dataUpdate(); } if( imu_call_5 >= 50 ) //20HZ { imu_call_5=0; if( debug_en == 1) { printf("mpuaccx = %.4f mpuaccy = %.4f mpuaccz = %.4f\r\n" , mpu6050.acc_normal.x , mpu6050.acc_normal.y,mpu6050.acc_normal.z); printf("hmc_normalx = %.4f hmc_normaly = %.4f hmc_normalz = %.4f\r\n" , hmc5883l.hmc_normal.x , hmc5883l.hmc_normal.y , hmc5883l.hmc_normal.z); printf("temperature = %.4f pressure = %.4f altitude = %.4f altitude_offset = %.4f\r\n" , ms611.temperature , ms611.pressure , ms611.altitude , ms611.altitude_offset); } } }

接下來就是頻率為100hz的系統數據更新函數,還有50hz的底層電機控制函數。

//motor_top.cpp
void MotorTop::motorTopCall(void) { motorPWMRenew( 1 , motor1.speedControl(expect_angle_speed_m[0] , board.getMotorEncoderCNT(1) ) ); motorPWMRenew( 2 , motor2.speedControl(expect_angle_speed_m[1] , board.getMotorEncoderCNT(2) ) ); if(motor_enable_num >= 3) { motorPWMRenew( 3 , motor3.speedControl(expect_angle_speed_m[2] , board.getMotorEncoderCNT(3) ) ); } if(motor_enable_num >= 4) { motorPWMRenew( 4 , motor4.speedControl(expect_angle_speed_m[3] , board.getMotorEncoderCNT(4) ) ); } }

此函數獲得此刻編碼器的速度數據之后先進行速度控制,實現如下。

//robot_wheel_top.cpp
float MotorControl::speedControl(float expect_speed , float unit_count) { //0.3 float filter = 0.3 * (50 * pid_t); expect_angle_speed = (1-filter) * measure_angle_speed + filter * expect_speed; if(motor_simulation_model == 1) { measure_unit_encoder = (simulation_max_angel_speed/360.0f) *( pid_parameters_.i_pidout/pwm_max ) * encoder_num * pid_t; } else { measure_unit_encoder = unit_count; //if you using real motor } //expect unit encoder num in one cycle to pid expect_unit_encoder = ( expect_angle_speed / 360 ) * encoder_num * pid_t; expect_total_encoder += expect_unit_encoder ; //recording total encoder measure_total_encoder += measure_unit_encoder ; //recording total angle for robot coordinate calculation d_past_angle += (measure_unit_encoder/encoder_num)*360; past_total_angle+=(measure_unit_encoder/encoder_num)*360; //calc motor speed degree/s measure_angle_speed = measure_unit_encoder * 360 / ( encoder_num*pid_t); //motor speed pid control function pidOrdinaryCall(expect_total_encoder , measure_total_encoder , expect_unit_encoder , measure_unit_encoder , pwm_max); return pid_parameters_.i_pidout; }

一開始,先對車輪速度進行滑動平局濾波,之后計算期望速度,實際速度,期望位置,實際位置。最后將數據送入PID函數,進行位置速度雙閉環控制。

之后是20hz的機器人底盤控制robotWheelTopCall()函數。

//robot_wheel_top.cpp
void RobotWheel::robotWheelTopCall(void)
{

    robotDataUpdate();
    chassisControl();    // control your robotic chassis
    robotCoordCalc();    // for robot localization
    headControl();       // control your robotic head
    armControl();        // control your robotic arm
}

首先機器人會將所有數據更新,包括全局速度(global_speed)、機器人速度(robot_speed)、之前測量的電機轉速,位置和期望電機轉速,位置、數字舵機的位置(如果有),四個機器人系統數據和IMU數據,最后還有pid數據(如果底層收到上位機的設定參數的請求,便會更新參數)。

然后調用ChassisControl()控制機器人底盤,它先分別判斷是否收到新的命令,如果收到便執行相應操作。

最后調用robotCoordCalc()通過電機測量的三個電機的速度與位置計算機器人坐標系的速度位置。

我已經將代碼同步到git上,里面有我改好的三輪底盤機器人地層程序。

git:https://github.com/63445538/Embedded/tree/my_pro

在/0_Project/STM32F4/Application_HF_Wheel_Robot_Beta/目錄下有已經建好的keil工程,應該沒問題。如果有問題請留言。

參考:

[1]R.西格沃特等.自主移動機器人導論[M].第二版.李人厚等譯.西安:西安交通大學出版社,2013.5

[2]https://github.com/HANDS-FREE

[3]http://www.runoob.com/cplusplus/cpp-classes-objects.html

[4]http://yfrobot.com/forum.php?mod=viewthread&tid=2411

總結

以上是生活随笔為你收集整理的[如何构建自己的轮式移动机器人系统&#183;从入门到放弃]机器人底层篇的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

亚洲国产精品久久 | 黄网站污 | 波多野结依在线观看 | 黄色com| 国产黄色免费 | 欧美巨乳网 | 国产亚洲欧洲 | 狠狠久久| 精品成人a区在线观看 | 久草久热 | 特黄免费av | 久久久久久久久久久福利 | 成人av免费在线播放 | 国产黄色精品在线观看 | www.黄色| 免费视频一区二区 | 国产无套精品久久久久久 | 在线日韩中文字幕 | 欧洲亚洲激情 | 91视频啪| ww视频在线观看 | 99热最新精品 | japanesefreesexvideo高潮 | 韩日精品在线观看 | 日韩av午夜在线观看 | 亚洲涩综合| 99高清视频有精品视频 | 亚洲精品99久久久久久 | 色噜噜色噜噜 | 中文字幕在线观看av | 天天干国产 | 一区二区 久久 | 国产中文欧美日韩在线 | 国产日韩精品一区二区在线观看播放 | 色网站在线观看 | 激情丁香5月 | av在线免费观看不卡 | 久久久久一区二区三区四区 | 国产破处在线播放 | 欧美大荫蒂xxx | 天天射综合网站 | 国产精品 久久 | 亚洲色图27p | 91网站观看 | 久久久久久久久久久久国产精品 | 一区在线免费观看 | 婷婷丁香导航 | 91试看| 国产精品一二 | 麻豆精品传媒视频 | 亚洲免费在线看 | 欧美乱淫视频 | 国产精选在线 | 国产一区二区不卡在线 | 国产在线黄 | 成人av一二三区 | 国产手机av在线 | 国产伦精品一区二区三区无广告 | 91精品一区二区三区蜜臀 | 日韩av专区 | 激情深爱.com | 综合天堂av久久久久久久 | 日韩电影中文,亚洲精品乱码 | 国产精品免费视频久久久 | 美女网站视频色 | 日韩在线免费小视频 | 成人aaa毛片 | 久久久久久久电影 | 亚洲精品毛片一级91精品 | 女人18毛片a级毛片一区二区 | 夜夜操夜夜干 | 99久视频 | 亚洲精品乱码久久久一二三 | 性色av一区二区三区在线观看 | 色瓜| 欧美久久久久久久久久 | 一级黄色在线视频 | 国产精品免费看 | 96av在线| 日韩午夜视频在线观看 | 九九热只有精品 | 成人黄色电影在线观看 | 天天看天天干天天操 | 久久久久国产精品免费免费搜索 | 日韩精品一区二区在线观看视频 | 国内外成人免费在线视频 | 久久观看免费视频 | 深爱激情开心 | 四虎在线免费观看 | 婷婷婷国产在线视频 | 亚洲91精品在线观看 | av三区在线 | 国产粉嫩在线 | 天天夜夜操 | 激情综合一区 | 精品视频免费久久久看 | 高清有码中文字幕 | 色香com. | 亚洲综合视频网 | 91av亚洲| 麻豆久久一区 | 久久精品99北条麻妃 | 亚洲综合五月 | 精品一区 在线 | 欧美日韩一区二区视频在线观看 | 免费国产亚洲视频 | 日韩在线观看一区二区三区 | 色吧av色av| 黄色在线小网站 | 亚洲三级在线播放 | 婷婷五月色综合 | 97国产情侣爱久久免费观看 | 亚洲精品免费在线 | 五月色综合 | 亚洲精品在线观 | a色网站| 日韩综合精品 | 日韩视频一区二区三区在线播放免费观看 | www久草| 国产色视频123区 | 国产精品美女久久久久久久久久久 | 免费观看十分钟 | 亚洲国产人午在线一二区 | 免费在线成人 | 五月天婷亚洲天综合网精品偷 | 久久久久成人精品免费播放动漫 | 成人午夜黄色影院 | 韩国视频一区二区三区 | 日日夜夜狠狠 | 天天操天天射天天舔 | 激情自拍av | 亚洲精品tv| 91网站在线视频 | 日韩视频一区二区三区在线播放免费观看 | 少妇高潮流白浆在线观看 | 最近中文国产在线视频 | 成人免费看片98欧美 | 爱干视频 | 欧美高清视频不卡网 | 日韩精品一区二区三区免费视频观看 | 欧美国产日韩久久 | 中文在线字幕免 | 免费在线观看毛片网站 | 99久久国产免费,99久久国产免费大片 | 一区 二区电影免费在线观看 | 深爱激情综合 | 黄色在线视频网址 | 粉嫩高清一区二区三区 | 99久久精品国产一区二区三区 | 亚洲永久国产精品 | 视频1区2区 | 亚洲视频免费 | 国产亚洲精品久久久久久无几年桃 | 精品久久久久久久久久久院品网 | 91高清视频在线 | av黄色国产| 久久伊人精品天天 | 中文字幕文字幕一区二区 | 日韩免费看片 | 久久精品成人 | 色婷婷伊人 | 成人av免费在线播放 | 又黄又刺激的视频 | 黄网站色视频免费观看 | 麻花豆传媒mv在线观看 | 丁香婷婷综合激情 | 97超级碰碰碰视频在线观看 | 99久久婷婷国产综合亚洲 | 欧美91精品国产自产 | 久久久久久久国产精品 | 天天插综合 | 349k.cc看片app| 久久久网页 | 日批视频在线观看免费 | 欧美精品在线视频观看 | 最近中文字幕高清字幕在线视频 | 亚洲免费成人 | 91精品国产92久久久久 | 97超碰在线免费观看 | 丁香婷婷在线 | 中文字幕乱视频 | 西西44人体做爰大胆视频 | 玖操 | 欧美成人h版电影 | 一区二区精品在线 | 欧美视频国产视频 | 国产精品久久久久久久久搜平片 | 国产资源av | 91桃色在线播放 | 免费黄色网址大全 | 久久久 精品 | 亚洲人成免费网站 | 久久99深爱久久99精品 | 成年人视频在线 | 久久久麻豆精品一区二区 | 探花系列在线 | 久久国产美女视频 | 日日天天av | 精品日韩中文字幕 | 丁香六月天婷婷 | 国产精品自产拍在线观看网站 | 免费日韩一级片 | 久久久久久免费毛片精品 | 五月婷婷在线综合 | a v在线视频| 99久久精品国产一区二区成人 | 2019中文在线观看 | 国产亚洲精品久久久久久久久久久久 | 久久综合爱 | 久久精品久久精品 | 久久精品久久国产 | 国产精品久久久电影 | 这里只有精彩视频 | 中文字幕有码在线 | 激情五月婷婷激情 | 欧美一二在线 | 免费日韩 精品中文字幕视频在线 | 97精品视频在线 | 91成品人影院 | 日本在线中文 | 精品日韩av | 97在线观看视频 | 韩国精品视频在线观看 | 97人人模人人爽人人少妇 | 日日夜夜综合网 | 国产黄在线观看 | 蜜臀av一区二区 | 久草在线看片 | 久久婷婷网 | 成人在线观看免费 | 欧美日韩综合在线 | 久久成人综合视频 | 国产婷婷一区二区 | 天天干天天操天天入 | 精品欧美一区二区三区久久久 | 999国内精品永久免费视频 | 综合中文字幕 | 亚洲精品456在线播放第一页 | 国产一区二区精品91 | 国产免费观看久久 | 又黄又刺激又爽的视频 | 国产剧在线观看片 | 少妇bbbb揉bbbb日本 | 一区二区三区电影大全 | 成人午夜影院在线观看 | 久久国产成人午夜av影院宅 | 99理论片 | 九九热在线精品视频 | av在线在线| 久久免费精品 | 国产精品一区二区三区四区在线观看 | 日韩精品一区二区电影 | 四虎影视成人 | 中文亚洲欧美日韩 | 911国产 | 国产精品久久久久久a | 天堂av观看| 高清国产一区 | 超碰官网 | 91九色porny蝌蚪视频 | 色综合久久网 | 日韩网站一区 | 国模精品在线 | 久久曰视频 | 一级欧美一级日韩 | 国产一区二区三区免费在线观看 | 欧美日韩伦理一区 | 中文字幕在线播出 | 日韩精品1区2区 | 国产成人在线观看 | 色99久久 | 亚洲精品在线看 | 一区二区三区在线免费播放 | 色999在线| av 一区二区三区四区 | 欧美一区二区三区激情视频 | 国产资源免费在线观看 | 精品在线99 | 极品中文字幕 | 免费在线观看一区 | 欧美日韩国产综合一区二区 | www麻豆视频 | 亚洲成人软件 | 亚洲影院色 | 中文字幕亚洲在线观看 | 国产一级黄色av | 五月天国产精品 | 精品国产成人 | 国产麻豆果冻传媒在线观看 | 波多野结衣小视频 | 成年人三级网站 | 天天操天操 | 伊人亚洲综合网 | a久久免费视频 | 国产在线精 | 爱干视频| 日韩三级av | 久久精品国产99国产 | 夜夜视频资源 | 日韩欧美高清视频在线观看 | 亚洲女同ⅹxx女同tv | 精品日韩在线一区 | 日日干 天天干 | 在线观看精品国产 | 成人91在线观看 | 婷婷日日| 一本一本久久a久久精品综合 | 国内久久视频 | 日韩av片无码一区二区不卡电影 | 国产精品扒开做爽爽的视频 | 四虎影视成人永久免费观看视频 | 亚洲女同videos | 久久99国产精品久久99 | 国产一区二区三区久久久 | 黄色毛片网站在线观看 | 欧美在线你懂的 | 丁香五月缴情综合网 | 国产 中文 日韩 欧美 | 亚洲色图美腿丝袜 | 91探花视频 | 亚洲精品国产电影 | 91久久丝袜国产露脸动漫 | 色婷婷伊人 | 韩国视频一区二区三区 | 久草在线中文视频 | 国产精品久久久久久久久久久不卡 | 日本婷婷色 | 91人人爱 | 粉嫩av一区二区三区四区在线观看 | 久久久www | 国产污视频在线观看 | 久久狠狠干 | 免费在线色电影 | 国产成人综合图片 | 91久久一区二区 | 成人全视频免费观看在线看 | 久久免费观看少妇a级毛片 久久久久成人免费 | 久久字幕 | 1区2区视频 | 天天插视频 | 久久香蕉电影 | 黄色网www | 一区二区丝袜 | 在线免费国产 | 成人资源在线播放 | 免费视频久久久久 | 天天在线视频色 | 天天天天天天干 | 97视频资源 | 成人国产精品 | 久久婷婷色 | 在线观看国产日韩欧美 | 国产午夜一区二区 | 欧美日韩高清免费 | 久精品视频免费观看2 | 亚洲免费精品一区二区 | 午夜黄网 | 国产精品永久久久久久久www | 精品一区中文字幕 | 色综合久久88色综合天天6 | 九热在线 | 色悠悠久久综合 | 日日夜夜添 | 天天草天天操 | av中文字幕免费在线观看 | 丁香花中文字幕 | 色婷婷激情网 | 国产精品久久久久久吹潮天美传媒 | 国产日韩视频在线 | 久久久久久久久久久久久久电影 | 在线观看亚洲专区 | 国产资源免费在线观看 | 欧美精品三级在线观看 | 久久久久免费网站 | 伊人久久婷婷 | 91精品免费视频 | 成人精品国产免费网站 | 91高清在线看 | 丁香激情五月 | 91视频免费网址 | 中文字幕亚洲综合久久五月天色无吗'' | 六月激情婷婷 | 99视频精品 | 日韩videos | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 亚洲九九九在线观看 | 国产精品视频免费在线观看 | 亚洲精品国产日韩 | 最近中文字幕mv免费高清在线 | 在线观看a视频 | 成片免费观看视频 | 99re视频在线观看 | 丁香激情综合久久伊人久久 | 黄色com| 午夜视频免费 | 国产免码va在线观看免费 | 国产精品一区二区三区在线播放 | 成人黄色免费在线观看 | 婷婷国产精品 | 国产精品国产亚洲精品看不卡 | 久久久首页 | 精品女同一区二区三区在线观看 | 超碰97在线人人 | 日日躁夜夜躁aaaaxxxx | 色婷婷亚洲 | 久久成人欧美 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 天天插日日射 | 亚洲免费视频观看 | 丁香婷婷久久久综合精品国产 | 黄色免费观看视频 | 在线天堂中文在线资源网 | 日本不卡一区二区三区在线观看 | 色噜噜日韩精品欧美一区二区 | 久久网址 | 日韩二区三区在线 | 日韩精品久久久免费观看夜色 | 全久久久久久久久久久电影 | 一本一本久久a久久精品综合 | 日韩一区二区免费视频 | 午夜av影院 | 久久国产亚洲精品 | 特片网久久 | 久久视频免费在线 | 久草观看 | 2019中文最近的2019中文在线 | 有码视频在线观看 | 在线一区观看 | 九九九九九精品 | 久久99日韩| 色99色| 激情婷婷av | 欧美日韩性视频 | www.伊人网| 亚洲乱码国产乱码精品天美传媒 | 国产婷婷一区二区 | 亚洲综合精品在线 | 国产99免费视频 | av片中文字幕 | 91麻豆精品国产自产在线 | 色婷婷国产 | 亚洲精品视频第一页 | 夜夜爽夜夜操 | 国产高清视频免费观看 | 在线成人小视频 | 8x8x在线观看视频 | 黄色影院在线免费观看 | 欧洲精品二区 | 亚洲精品18日本一区app | www天天干com| 一区二区三区韩国免费中文网站 | 黄色av一级片 | 99热精品国产 | 久久久免费视频播放 | 欧美成人亚洲 | 精品久久久久久久久久久久 | 亚洲美女视频在线 | 国产精品a成v人在线播放 | 人人插超碰 | 日韩精品一区二区三区电影 | 一级黄色片在线免费观看 | 天天综合色天天综合 | 四虎国产精品免费观看视频优播 | 91系列在线观看 | 草莓视频在线观看免费观看 | 在线观看色网站 | 91亚洲精品国产 | 精品久久1| 97在线观 | 欧美性色xo影院 | 国产精品乱码一区二三区 | 国产精品福利视频 | 中文字幕色站 | 国产精品99久久免费黑人 | 伊人天堂网| 精品在线视频播放 | 国产精品va在线播放 | 日韩精品一区电影 | 在线观看中文字幕视频 | 国产亚洲精品久久久久久网站 | 最近中文字幕mv免费高清在线 | 国产婷婷视频在线 | 中文区中文字幕免费看 | 欧美肥妇free | 最近中文字幕第一页 | 在线国产一区 | 操操操天天操 | 精品一二 | 国产片免费在线观看视频 | 久久久久久久久久伊人 | 色激情五月 | 中文字幕视频一区二区 | www日韩视频 | 狠狠色丁香婷婷综合基地 | 91香蕉视频色版 | 三级视频片 | 日本韩国中文字幕 | 亚洲高清av在线 | 中文字幕在线一区观看 | 天天干夜夜干 | 91成人区| 成人理论电影 | 国产成人精品三级 | 中文字幕888 | 日日摸日日碰 | 九九日九九操 | 婷婷香蕉| 久久黄色免费 | 一级黄色片网站 | 日韩av一区二区在线影视 | 午夜精品一区二区三区在线播放 | 91久久久久久久一区二区 | 欧美成人h版电影 | 国产资源在线免费观看 | 亚洲欧美日韩一二三区 | 国产人成精品一区二区三 | 久久久久久久久久久影院 | 亚洲综合精品视频 | 精品欧美一区二区精品久久 | 国产精品久久99综合免费观看尤物 | 97香蕉视频 | 亚洲国产成人精品在线观看 | 国产一区二区视频在线播放 | 欧美极品在线播放 | 一区二区中文字幕在线观看 | 美女免费视频网站 | caobi视频 | 看av免费网站 | 在线观看一 | 日韩大片在线免费观看 | 西西人体www444 | 在线免费观看涩涩 | 国产破处精品 | 国产高清在线不卡 | 91九色国产蝌蚪 | 欧美黑人xxxx猛性大交 | www.日日操.com| 狠狠干夜夜操 | 日韩xxxxxxxxx | 97韩国电影 | 美女网站视频一区 | 久久人人爽人人爽 | 人人爽人人爽人人片 | 日韩国产欧美在线播放 | 亚洲国产一二三 | 亚洲一级黄色片 | 成人国产精品电影 | 欧美日韩二区三区 | 亚洲一区二区精品3399 | 四虎国产精品免费 | www色com| 欧美性生活久久 | 日韩精品中文字幕一区二区 | 国产精品一区二区三区视频免费 | 久久精品99久久久久久2456 | 91探花国产综合在线精品 | 免费av在线网站 | 天天艹天天操 | 91色一区二区三区 | 亚洲精品一区二区18漫画 | 国产流白浆高潮在线观看 | 国产在线中文字幕 | 日韩欧美精品在线视频 | 免费合欢视频成人app | 国产高h视频 | 日韩一级黄色大片 | 久久久久观看 | 一区二区伦理电影 | 亚洲黄色一级视频 | 最新av中文字幕 | 亚洲精品在线国产 | av电影免费在线看 | 激情久久久久久久久久久久久久久久 | 国产成人在线一区 | 国产精品免费观看久久 | 五月婷婷在线视频观看 | 国产91丝袜在线播放动漫 | 五月婷婷综合激情 | 91看片成人 | 亚洲aⅴ一区二区三区 | 激情五月色播五月 | 国产成人在线网站 | 国产精品99久久久久久小说 | 日本激情视频中文字幕 | 久久天堂精品视频 | 亚洲欧美偷拍另类 | 国产一级二级三级视频 | 波多野结衣在线观看一区二区三区 | 啪啪免费试看 | 精品99999 | 精品欧美在线视频 | 天天曰夜夜爽 | 黄色日本免费 | 99久久精品久久久久久清纯 | 久久久国产一区二区三区 | 中文字幕在线第一页 | 久草在线免费电影 | 日本久久影视 | 337p日本欧洲亚洲大胆裸体艺术 | 婷婷丁香花五月天 | 亚洲永久精品在线 | 97超碰国产精品 | 日韩特级片| 日日摸日日爽 | 国产精品久久久999 国产91九色视频 | 免费a级毛片在线看 | 黄p网站在线观看 | 在线三级av| 最近更新的中文字幕 | www国产一区 | 麻豆久久精品 | 香蕉久久久久久久 | 国产高清在线a视频大全 | 国产高清不卡一区二区三区 | 日韩伦理片hd | 亚洲精品视频久久 | 久草视频手机在线 | 黄色av电影在线 | 精品福利av | 91看片淫黄大片在线播放 | 亚洲成a人片在线观看中文 中文字幕在线视频第一页 狠狠色丁香婷婷综合 | 在线一二区 | 超碰在线亚洲 | 天干啦夜天干天干在线线 | 久久国产精品一区二区三区 | 97精品一区二区三区 | 中文字幕在线观看网站 | 中文乱幕日产无线码1区 | 成人国产网址 | 丁香九月婷婷综合 | 三级黄色免费片 | 色在线最新| av 在线观看 | 一区在线观看视频 | 亚洲精品视频在线播放 | 国产精品第52页 | 久久九九久久 | 99久久www| 欧美专区日韩专区 | 国产一区高清在线观看 | 久久久久久亚洲精品 | 久久艹欧美 | 欧美日韩调教 | 在线观看中文 | 91看片麻豆 | 97成人精品 | 色综合天天综合 | 99视频精品免费视频 | 少妇av片 | 亚洲欧美视频在线 | 97视频免费观看 | 亚洲a在线观看 | 91成人网页版 | 国产在线无 | 免费观看久久 | 丁香六月婷婷开心婷婷网 | 久久精品在线视频 | 婷婷色 亚洲| 日本激情视频中文字幕 | 色吊丝在线永久观看最新版本 | 一区二区视频电影在线观看 | 成人在线你懂得 | 亚洲天堂网视频在线观看 | 色丁香色婷婷 | 国产99久久久精品视频 | 日韩av免费一区二区 | 在线视频 精品 | 久草在线免费资源站 | 91重口视频| 97视频一区| 一级片视频在线 | 欧美另类交人妖 | 91亚洲国产成人久久精品网站 | 91av欧美| 亚洲日本va午夜在线影院 | 久久综合五月天婷婷伊人 | 国产成人精品在线观看 | 日韩中文字幕免费视频 | 日韩毛片在线播放 | 91手机视频 | 欧美日韩精品在线观看视频 | 激情视频在线观看网址 | 欧美天天干 | 亚洲人人精品 | 五月天激情综合网 | 久久精品中文字幕 | 成人免费观看视频大全 | 成人午夜久久 | 久久久久久美女 | 99久久精品久久久久久动态片 | 97超碰资源网 | 99久久日韩精品免费热麻豆美女 | 亚洲乱码中文字幕综合 | 色综合夜色一区 | 亚洲另类交| 色搞搞| 久久a国产| 最新av电影网站 | 国产热re99久久6国产精品 | 国产黄色网 | 久久久久久毛片 | 中文字幕视频观看 | 亚洲一区美女视频在线观看免费 | 狂野欧美激情性xxxx | 亚洲精品国产精品国自产 | 久久久久国产免费免费 | 亚洲激情一区二区三区 | 日韩色一区二区三区 | 一区二区三区动漫 | 婷婷国产精品 | 精品欧美乱码久久久久久 | 午夜视频在线观看网站 | 青青河边草观看完整版高清 | 九九色综合 | 黄色大片视频网站 | 在线精品国产 | 国产精品久久久久久久久久久久 | 日韩中文字幕免费在线播放 | 亚洲我射av | 91pony九色丨交换 | 亚洲精品av在线 | 久草在线中文888 | 99激情网 | 国产成人久久 | av噜噜噜在线播放 | 天天操天天舔天天干 | 国产精品三级视频 | 91久久精品一区二区三区 | 久久99网站| 免费视频97 | 91香蕉视频色版 | 国产成人精品电影久久久 | 久久好看 | 亚洲综合色视频在线观看 | 亚洲性xxxx| 国产系列在线观看 | 日本久久免费电影 | 在线黄频| 欧美日韩在线观看一区二区三区 | 久久99国产精品二区护士 | 婷婷伊人五月天 | 国产精品视频全国免费观看 | 国产亚洲精品久久久久久无几年桃 | 在线观看免费一区 | 日韩欧美精选 | 日韩欧美在线观看一区二区 | 91av视频在线播放 | 十八岁以下禁止观看的1000个网站 | 97成人在线视频 | 亚洲人成人在线 | 精品中文字幕在线 | 一二三久久久 | 久草在线费播放视频 | 九九热免费精品视频 | av在线网站免费观看 | av成人在线网站 | 亚洲成人av免费 | 国产精品h在线观看 | 美女视频免费一区二区 | 去看片| 免费a视频在线 | 色天天 | 国产欧美最新羞羞视频在线观看 | 国产手机av | 国产黄色一级片 | 亚洲精品永久免费视频 | 一二三精品视频 | 国产黄色片一级三级 | 久久伊人精品一区二区三区 | 亚洲成av人片 | 精品黄色在线观看 | 国产精品久久久久久久久软件 | 操操操干干干 | 国产精品日韩久久久久 | 91看毛片 | 中文字幕在线有码 | 91九色国产视频 | 亚洲高清免费在线 | 成人在线超碰 | 天堂va在线高清一区 | 成人黄色在线电影 | 久草精品视频在线观看 | 激情视频区| 国产精品免费观看国产网曝瓜 | 丁香婷婷激情 | 久久99爱视频 | www.一区二区三区 | 久草在线在线精品观看 | 免费国产在线精品 | 亚洲一区二区精品在线 | 激情久久小说 | 国产精品视频99 | 日日弄天天弄美女bbbb | 亚洲综合在线播放 | 国产精品一区二区久久精品爱涩 | 国产一级免费视频 | 欧美国产日韩一区二区 | 91亚洲精品国偷拍自产在线观看 | 激情欧美xxxx | 毛片的网址 | 亚洲久草视频 | 国产中出在线观看 | 日韩一区二区三区在线观看 | 日韩欧美有码在线 | 人人爽人人爽人人片 | www免费看 | 久久精品精品 | 国产一区二区三区网站 | av中文字幕在线播放 | 亚洲国产高清在线观看视频 | 中文字幕成人在线观看 | 国产精品剧情在线亚洲 | 色婷婷伊人 | av中文字幕亚洲 | 九九热精品国产 | 精品一区二区在线看 | 久久男人视频 | 51久久夜色精品国产麻豆 | 亚洲涩涩网| 国产成人精品一区二区在线观看 | 在线视频久久 | 91精品国产麻豆国产自产影视 | 亚洲精品99久久久久中文字幕 | 一二三区视频在线 | 热久久99这里有精品 | 天天射天天干天天操 | 99九九热只有国产精品 | 99热高清| 日韩中文字幕电影 | 爱色婷婷| 欧美日韩国产伦理 | 久久在线精品 | 综合激情网 | 国产99久久九九精品 | 亚洲春色综合另类校园电影 | 97人人添人澡人人爽超碰动图 | 国产韩国日本高清视频 | 伊人看片 | 激情婷婷丁香 | 狠狠色噜噜狠狠 | 久久激情日本aⅴ | 成年人在线 | 天天干天天做 | 亚洲综合成人在线 | 免费男女羞羞的视频网站中文字幕 | 欧美91在线 | 国产精品久久一区二区三区, | 99久久精品免费一区 | 国产精品女教师 | 91av视频免费在线观看 | 日韩亚洲国产中文字幕 | 国产又粗又硬又爽视频 | 日本三级久久 | 青春草视频| www.狠狠干| 国产成人黄色 | 在线观看视频一区二区三区 | 国产精品久久婷婷六月丁香 | 日韩极品视频在线观看 | av观看免费在线 | 91精品电影 | 久久亚洲专区 | 中文字幕 成人 | 日韩欧美精品在线视频 | 欧美久草视频 | 国产99久久精品一区二区300 | 四虎成人在线 | 中文字幕在线免费看 | 9999在线 | 免费在线观看国产精品 | 四虎影视8848dvd| 在线亚洲精品 | 99久久9| 97在线播放视频 | 久久久午夜精品理论片中文字幕 | 五月色丁香 | 五月婷婷视频在线 | 在线观看91精品国产网站 | 日韩偷拍精品 | 国产亚洲精品久久久久久电影 | 日韩精品欧美专区 | 99色免费视频 | 日韩av一区二区三区 | 成人永久在线 | 久久精品视频国产 | 中文字幕视频观看 | 最新免费av在线 | 日本黄区免费视频观看 | 国产精品久久久久久模特 | 免费精品国产va自在自线 | 视频直播国产精品 | 欧美一性一交一乱 | www久久久 | 最新av网址在线 | 青草视频在线免费 | 四虎国产精品成人免费影视 | 免费高清看电视网站 | 国产一区二区三区免费视频 | 91福利国产在线观看 | 麻豆视频免费在线观看 | 天天天色综合a | 色婷婷88av视频一二三区 | 韩日在线一区 | 亚洲精品综合一二三区在线观看 | 外国av网| 国产剧情一区二区 | 亚洲午夜小视频 | 日韩精品一区二区三区不卡 | 国产91精品看黄网站在线观看动漫 | 国产精品亚洲综合久久 | 一区二区三区观看 | 偷拍精偷拍精品欧洲亚洲网站 | 狠狠的操狠狠的干 | 欧美日韩中文字幕综合视频 | 午夜黄色影院 | 一区 二区电影免费在线观看 | 国产精品资源在线 | 97视频免费在线看 | 久久久久久久久久久久亚洲 | 成人一区二区三区中文字幕 | 97视频在线免费播放 | 91亚色视频 | 最新精品国产 | 国产麻豆精品在线观看 | 天天操天天爽天天干 | 三级黄色在线观看 | 国产在线传媒 | 国产精品99精品 | 亚洲美女视频在线观看 | 一级黄色大片在线观看 | av在线电影网站 | 99国产视频在线 | 国产精品白丝av | 免费看av在线| 亚洲精品在线一区二区三区 | 中文字幕在线观看第一区 | 天堂av在线中文在线 | 亚洲激情精品 | 午夜精品一区二区三区在线观看 | 4p变态网欧美系列 | 1000部18岁以下禁看视频 | 亚洲天天摸日日摸天天欢 | 成人免费视频网站在线观看 | 婷婷久久五月 | 精品毛片一区二区免费看 | 免费久久精品视频 | 友田真希x88av| 日韩精品影视 | 亚洲伊人色 | 最近日韩中文字幕中文 | 成人黄色片免费 | 久久男人视频 | 色就是色综合 | 69久久久| 日韩黄色免费 | 日本久久免费电影 | 伊人影院av | 国产中文字幕一区二区 | 国内精品亚洲 | 国产在线观看免费观看 | 日日躁你夜夜躁你av蜜 | 成人在线视频观看 | 久久 精品一区 | 国产午夜一区二区 | 国产91探花 | 国产首页| 人人草人 | 91麻豆精品国产91久久久无需广告 | 人人爽人人插 | 久久精品中文视频 | 毛片区 | 日韩毛片在线一区二区毛片 | 精品 一区 在线 | www.888av| 黄色a视频| 中文字幕亚洲国产 | 久久久99精品免费观看 | 免费av在线网 | 亚洲国产精品日韩 | 国产美女在线精品免费观看 | 久久99久国产精品黄毛片入口 | 欧美激情精品 | 99视频在线免费 | 91免费视频黄 | 国产精品久久久精品 | 久久久久女人精品毛片九一 | 日韩女同av | 91经典在线| 99久久久久久国产精品 | 日韩一二区在线 | aa一级片 | 在线看片91| 国产精品久久久久久久久久久杏吧 | 国产高清av在线播放 | 日韩成人免费电影 | 欧美激情综合五月色丁香小说 | 日韩网站在线观看 | 国产日女人 |