GifuRobo B term
Dependencies: FastPWM cal_PID mbed omuni
Fork of omuni_speed_pid by
main.cpp
- Committer:
- Akito914
- Date:
- 2017-09-19
- Revision:
- 7:5cdd805ca8a3
- Parent:
- 6:22ad12682a5d
- Child:
- 8:956a76b98da0
File content as of revision 7:5cdd805ca8a3:
#include "mbed.h" #include "omuni.hpp" #include "cal_PID.hpp" #include "timeBaseTrapezoidalMotionCal.h" #include "FastPWM.h" #include "soundData.h" #define LIFT_METER_INVERSION 1 const int comTimeout_ms = 200; const float omuni_speed[4] = {0, 1.0, 2.0, 3.0}; const float omuni_speed_max = 2.0f; const float omuni_burst_coeff = 1.5f; const float omega_max = 0.7 * 2 * 3.14159265f; const float omega_f = 0.7 * 2 * 3.14159265f; const float wrap_radius = 1.0f; const float wrap_speed = 1.8f; const int8_t armDuty[] = {-100, -100}; const signed int spearDuty = 127; //const int32_t lift_preset_min = 200; const int32_t lift_preset_min = 320; // Temporary use const int32_t lift_preset_max = 480; const int32_t lift_preset[4] = { lift_preset_min, (lift_preset_max - lift_preset_min) / 3.0 + lift_preset_min, (lift_preset_max - lift_preset_min) * 2.0 / 3.0 + lift_preset_min, lift_preset_max, }; // 12bit 0 ~ 4095 const int32_t lift_max = lift_preset_max; const int32_t lift_min = lift_preset_min; const int32_t lift_up_speed = 10; const int32_t lift_down_speed = 10; const int32_t lift_prescaler = 10; const int omuniAddr[] = {0x10, 0x12, 0x14}; // 0000 , 1000 , 0100 const int armAddr[] = {0x16, 0x18}; // 1100 , 0010 const int spearAddr = 0x1a; // 1010 const int liftAddr = 0x1c; // 0110 const int EMO_Addr = 0x1e; // 1110 int comTimeout_count = 0; bool comTimeout_state = true; bool ems = false; bool arm = false; bool spear = false; int lift_targ = lift_preset[0]; int lift_stepTarg; bool lift_stepMoving = false; bool first_receive = true; int past_lift_gray = 0b00; int lift_inc = 0; const float soundGain = 1.0; int soundDataCount = 0; bool soundEnd = false; DigitalIn button(USER_BUTTON); DigitalOut led(LED1); Serial pc(USBTX, USBRX); Serial com(PA_11, PA_12); I2C i2cMaster(D14, D15); omuni omuni(&i2cMaster, TIM1, TIM2, TIM3, 533, 2.0f, omuniAddr, 0.4704f, 0.1f); AnalogIn lift_meter(PC_0); DigitalIn spear_sensor(PC_3); Timer mesPeriod; cal_pid lift_pid; Ticker comTimeout; FastPWM soundOut(PB_6); Ticker soundRenew; float speed_x, speed_y, omega; bool f; typedef union{ struct{ unsigned startByte :8; unsigned joyLX :8; unsigned joyLY :8; unsigned joyRX :8; unsigned joyRY :8; unsigned L2 :8; unsigned R2 :8; unsigned arrowL :1; unsigned arrowR :1; unsigned arrowU :1; unsigned arrowD :1; unsigned L1 :1; unsigned R1 :1; unsigned SELECT :2; unsigned lift_gray :2; unsigned spear :1; unsigned arm :1; unsigned pushL :1; unsigned pushR :1; unsigned EMO :2; unsigned checksum :8; }; uint8_t list[10]; }comBuf_t; void received_processing(comBuf_t *buf); int byteSum(int8_t byte); bool checksum_check(comBuf_t *buf); int drive_motor(int address,signed int duty); void emergencyStop(); comBuf_t comBuf; int grayDiffer2bit(int gray, int origin){ switch(origin){ case 0b00: switch(gray){ case 0b01: return 1; case 0b10: return -1; default : return 0; } case 0b01: switch(gray){ case 0b11: return 1; case 0b00: return -1; default : return 0; } case 0b11: switch(gray){ case 0b10: return 1; case 0b01: return -1; default : return 0; } case 0b10: switch(gray){ case 0b00: return 1; case 0b11: return -1; default : return 0; } } return 0; } bool checksum_check(comBuf_t *buf){ int sum = 0; for(int count = 0; count < 9; count++){ sum += buf->list[count]; } return (sum & 0b01111111) == buf->checksum; } void com_rx() { static int byteCount = 0; char temp = com.getc(); if(temp == 0b10000000){ comBuf.list[0] = temp; byteCount = 1; } else if((temp & 0b10000000) != 0){ // 想定外のデータ byteCount = 0; } else if(byteCount > 0){ comBuf.list[byteCount] = temp; if(byteCount < 9)byteCount += 1; else{ // データは揃った if(checksum_check(&comBuf)){ led = !led; comTimeout_count = 0; comTimeout_state = false; received_processing(&comBuf); } else{ byteCount = 0; } } } } int getJoy7bit(int raw){ return raw - 64; } void received_processing(comBuf_t *buf){ int lift_nearest_num = 0; ems = buf->EMO != 0; speed_x = omuni_speed_max * getJoy7bit(buf->joyLX) / 64.0f; speed_y = omuni_speed_max * getJoy7bit(buf->joyLY) / 64.0f; if(buf->pushL == 1){ speed_x *= omuni_burst_coeff; speed_y *= omuni_burst_coeff; } if(abs(getJoy7bit(buf->joyRY)) < 20 && abs(getJoy7bit(buf->joyRX)) > 50){ f = 1; omega = omega_f; if(getJoy7bit(buf->joyRX) < 0)omega *= -1; } else{ int diff = (int)buf->R2 - (int)buf->L2; omega = omega_max * diff / 127.0f; f = 0; } /* else if(buf->L2 >= 5 && buf->R2 < 5){ omega = omega_max * buf->L2 * -1.0f / 127.0f; f = 0; } else if(buf->L2 < 5 && buf->R2 >= 5){ omega = omega_max * buf->R2 * 1.0f / 127.0f; f = 0; } else{ omega = 0; f = 0; } */ if(buf->R1 || buf->L1){ speed_x = wrap_speed; speed_y = 0; f = 0; omega = wrap_speed / wrap_radius; speed_x *= buf->L1 ? 1 : -1; omega *= buf->L1 ? 1 : -1; } arm = buf->arm != 0; spear = buf->spear != 0; if(first_receive){ past_lift_gray = buf->lift_gray; first_receive = false; } if(buf->lift_gray != past_lift_gray){ // 段階移動 int lift_currentTarg; int lift_diff = 0; int lift_nearest = 10000; int lift_nextTarg; lift_currentTarg = lift_targ; if(lift_stepMoving){ lift_nextTarg = lift_stepTarg; } else{ lift_nextTarg = lift_currentTarg; } lift_diff = grayDiffer2bit(buf->lift_gray, past_lift_gray); for(int num = 0; num < 4; num++){ // 最寄りの段階 if(lift_nearest > abs(lift_preset[num] - lift_currentTarg)){ lift_nearest_num = num; lift_nearest = abs(lift_preset[num] - lift_currentTarg); } } if(lift_diff != 0){ if(lift_diff == 1){ if(lift_preset[lift_nearest_num] <= lift_currentTarg){ if(lift_nearest_num < 3)lift_nextTarg = lift_preset[lift_nearest_num + 1]; } else{ lift_nextTarg = lift_preset[lift_nearest_num]; } } else if(lift_diff == -1){ if(lift_preset[lift_nearest_num] >= lift_currentTarg){ if(lift_nearest_num > 0)lift_nextTarg = lift_preset[lift_nearest_num - 1]; } else{ lift_nextTarg = lift_preset[lift_nearest_num]; } } lift_stepTarg = lift_nextTarg; lift_stepMoving = true; } } past_lift_gray = buf->lift_gray; if(abs(getJoy7bit(buf->joyRY)) > 50 && abs(getJoy7bit(buf->joyRX)) < 20){ if(getJoy7bit(buf->joyRY) > 0)lift_inc = -5; else lift_inc = 5; } else{ lift_inc = 0; } } void soundRenew_intr(){ if(soundDataCount < soundData_size - 1000){ soundOut.write((soundData[soundDataCount] * soundGain / 256.0) + 0.5); soundDataCount++; } else{ soundEnd = true; } } void comTimeout_intr(){ if(comTimeout_count >= comTimeout_ms){ speed_x = 0; speed_y = 0; omega = 0; arm = 0; spear = 0; lift_inc = 0; comTimeout_state = true; } else{ comTimeout_count += 1; } } void arm_control(){ char armData[2] = {0}; armData[0] = arm? armDuty[0] : 0 ; armData[1] = arm? armDuty[1] : 0 ; i2cMaster.write(armAddr[0], &armData[0], 1); i2cMaster.write(armAddr[1], &armData[1], 1); } void lift_control(){ static int prescaler_count = 0; int32_t meter = 0; signed int duty = 0; int32_t error = 0; if(prescaler_count >= lift_prescaler){ prescaler_count = 0; error = lift_stepTarg - lift_targ; if(lift_stepMoving){ if(error > lift_up_speed){ lift_targ += lift_up_speed; } else if(error < lift_down_speed * -1){ lift_targ -= lift_down_speed; } else{ lift_targ = lift_stepTarg; lift_stepMoving = false; } } else{ if(lift_inc < 0){ if(lift_min < lift_targ + lift_inc)lift_targ += lift_inc; } else if(lift_inc > 0){ if(lift_max > lift_targ + lift_inc)lift_targ += lift_inc; } } } else{ prescaler_count += 1; } meter = lift_meter.read_u16() >> 6; #if LIFT_METER_INVERSION meter = 1023 - meter; #endif duty = lift_pid.get_pid(meter, lift_targ, 1); drive_motor(liftAddr, duty); } void spear_control(){ const int spear_timeout_short = 100; const int spear_timeout_long = 500; static int spear_state = 0; static int spear_timeout_count = 0; signed int duty = 0; int sensor = spear_sensor; // リミット時に1 switch(spear_state){ case 0 : // 待ち if(spear){ spear_state = 1; } else{ if(sensor == 0){ duty = spearDuty * -1; } } break; case 1 : // 初期位置から抜け出す duty = spearDuty; if(sensor != 0){ if(spear_timeout_count < spear_timeout_short){ spear_timeout_count += 1; } else{ spear_timeout_count = 0; spear_state = 0; } } else{ spear_timeout_count = 0; spear_state = 2; } break; case 2 : // 到達位置まで動かす if(sensor == 0){ duty = spearDuty; if(spear_timeout_count < spear_timeout_long){ spear_timeout_count += 1; } else{ spear_timeout_count = 0; spear_state = 3; } } else{ spear_timeout_count = 0; spear_state = 3; } break; case 3 : // 到達位置から抜け出す duty = spearDuty * -1; if(sensor == 0){ spear_state = 4; } break; case 4 : // 初期位置まで戻る if(sensor == 0){ duty = spearDuty * -1; } else{ spear_state = 0; } break; default : break; } drive_motor(spearAddr, duty); } void control(){ drive_motor(EMO_Addr, 1); omuni.set_speed(speed_x, speed_y, omega, f); omuni.drive(); arm_control(); lift_control(); spear_control(); } int main() { int period_us = 0; int count = 0; pc.baud(115200); com.baud(115200); pc.printf("Hello!\n"); com.attach(com_rx, Serial::RxIrq); i2cMaster.frequency(400000); omuni.set_speed(0.0f, 0.0f); omuni.set_pid(0, 3.0f, 0.07f, 0.05f); //omuni.set_pid(0, 6.0f, 0.14f, 0.10f); omuni.set_pid(1, 3.0f, 0.07f, 0.05f); omuni.set_pid(2, 3.0f, 0.07f, 0.05f); lift_pid.param(1.5, 0.0, 0.0); lift_pid.period(0.001); lift_pid.output(-127, 127); comTimeout.attach_us(comTimeout_intr, 1000); led = 0; /* soundOut.period_us(10); for(count = 0; count < 500; count++){ wait_ms(1); soundOut.write(count / 100.0); } soundRenew.attach_us(soundRenew_intr, 63); while(soundEnd == false); */ while(1) { wait(0.001); if(comTimeout_state == false){ control(); } else{ drive_motor(EMO_Addr, 0); } if(ems){ emergencyStop(); } } } int drive_motor(int address,signed int duty){ /* アドレスを指定してモータを駆動 */ char send_data; int ack; if((duty>127)||(duty<-128))return -1; /* 範囲外なら送信しない */ send_data=duty; ack=i2cMaster.write(address,&send_data,1); return ack; } void emergencyStop(){ drive_motor(omuniAddr[0], 0); drive_motor(omuniAddr[1], 0); drive_motor(omuniAddr[2], 0); drive_motor(armAddr[0], 0); drive_motor(armAddr[1], 0); drive_motor(liftAddr, 0); drive_motor(spearAddr, 0); drive_motor(EMO_Addr, 0); while(1); }