การใช้ Arduino Uno สำหรับการวางตำแหน่ง XYZ ของแขนหุ่นยนต์ 6 DOF: 4 ขั้นตอน
การใช้ Arduino Uno สำหรับการวางตำแหน่ง XYZ ของแขนหุ่นยนต์ 6 DOF: 4 ขั้นตอน
Anonim
Image
Image

โปรเจ็กต์นี้เกี่ยวกับการนำร่าง Arduino ที่สั้นและค่อนข้างง่ายไปใช้ในการจัดตำแหน่งจลนศาสตร์ผกผันของ XYZ ฉันได้สร้างแขนหุ่นยนต์เซอร์โว 6 ตัวแล้ว แต่เมื่อต้องหาซอฟต์แวร์เพื่อใช้งาน ไม่มีอะไรมากนอกจากโปรแกรมแบบกำหนดเองที่ทำงานบนเซอร์โวชิลด์แบบกำหนดเอง เช่น SSC-32(U) หรือโปรแกรมและแอปอื่นๆ ซับซ้อนในการติดตั้งและสื่อสารกับแขน จากนั้นฉันก็พบว่า "Robotic Arm Inverse Kinematics บน Arduino" ที่ยอดเยี่ยมที่สุดของ Oleg Mazurov ซึ่งเขาใช้จลนศาสตร์ผกผันในแบบร่าง Arduino อย่างง่าย

ฉันทำการแก้ไขสองครั้งเพื่อปรับโค้ดของเขา:

1. ฉันใช้ไลบรารี VarSpeedServo แทนไลบรารีเซอร์โวชิลด์แบบกำหนดเอง เพราะจากนั้นฉันก็สามารถควบคุมความเร็วของเซอร์โวได้ และฉันจะไม่ต้องใช้เซอร์โวชิลด์ที่เขาใช้ สำหรับใครก็ตามที่พิจารณาเรียกใช้โค้ดที่ให้ไว้ที่นี่ ฉันแนะนำให้คุณใช้ไลบรารี VarSpeedServo นี้ แทนที่จะเป็นไลบรารี servo.h เพื่อให้คุณสามารถชะลอการเคลื่อนไหวของแขนหุ่นยนต์ในระหว่างการพัฒนา หรือคุณอาจพบว่าแขนจะกระตุ้นคุณโดยไม่คาดคิด หน้าหรือแย่ลงเพราะจะเคลื่อนที่ด้วยความเร็วเซอร์โวเต็มที่

2. ฉันใช้เซนเซอร์/เซอร์โวชิลด์อย่างง่ายเพื่อเชื่อมต่อเซอร์โวกับ Arduino Uno แต่ไม่จำเป็นต้องใช้ไลบรารีเซอร์โวพิเศษเพราะใช้เพียงหมุดของ Arduino มีค่าใช้จ่ายเพียงไม่กี่เหรียญ แต่ไม่จำเป็น ทำให้การเชื่อมต่อเซอร์โวกับ Arduino เป็นไปอย่างราบรื่น และฉันจะไม่กลับไปใช้เซอร์โวเดินสายต่อกับ Arduino Uno อีกต่อไป หากคุณใช้เซนเซอร์/เซอร์โวชิลด์ คุณต้องทำการปรับเปลี่ยนเล็กน้อยซึ่งผมจะสรุปไว้ด้านล่าง

รหัสใช้งานได้ดีและช่วยให้คุณควบคุมแขนได้โดยใช้ฟังก์ชันเดียวที่คุณส่งผ่านพารามิเตอร์ x, y, x และความเร็ว ตัวอย่างเช่น:

set_arm(0, 240, 100, 0, 20); // พารามิเตอร์คือ (x, y, z, มุมกริปเปอร์, ความเร็วเซอร์โว)

ล่าช้า (3000); // จำเป็นต้องมีการหน่วงเวลาเพื่อให้เวลาแขนเคลื่อนที่ไปยังตำแหน่งนี้

ไม่สามารถง่ายกว่านี้ ฉันจะรวมร่างด้านล่าง

วิดีโอของ Oleg อยู่ที่นี่: การควบคุม Robotic Arm ด้วย Arduino และ USB Mouse

โปรแกรม คำอธิบาย และทรัพยากรดั้งเดิมของ Oleg: Inverse Kinematics ของ Oleg สำหรับ Arduino Uno

ฉันไม่เข้าใจคณิตศาสตร์ทั้งหมดที่อยู่เบื้องหลังกิจวัตร แต่ข้อดีคือคุณไม่ต้องใช้รหัส หวังว่าคุณจะลองดู

ขั้นตอนที่ 1: การปรับเปลี่ยนฮาร์ดแวร์

การปรับเปลี่ยนฮาร์ดแวร์
การปรับเปลี่ยนฮาร์ดแวร์

1. สิ่งเดียวที่จำเป็นคือเซอร์โวของคุณหมุนไปในทิศทางที่คาดไว้ ซึ่งอาจทำให้คุณต้องย้อนกลับการติดตั้งเซอร์โวของคุณ ไปที่หน้านี้เพื่อดูทิศทางเซอร์โวที่คาดหวังสำหรับเซอร์โวฐาน ไหล่ ข้อศอก และข้อมือ:

2. หากคุณใช้ชิลด์เซ็นเซอร์ที่ฉันใช้อยู่ คุณต้องทำสิ่งหนึ่งกับมัน: งอพินที่เชื่อมต่อ 5v จากชิลด์กับ Arduino Uno ให้พ้นทางเพื่อไม่ให้เชื่อมต่อกับบอร์ด Uno คุณต้องการใช้แรงดันไฟภายนอกบนชิลด์เพื่อจ่ายไฟให้กับเซอร์โวของคุณเท่านั้น ไม่ใช่ Arduino Uno หรือมันอาจทำลาย Uno ได้ ฉันรู้ในขณะที่ฉันเผาบอร์ด Uno สองแผงเมื่อแรงดันไฟฟ้าภายนอกของฉันอยู่ที่ 6 โวลต์มากกว่า 5 วิธีนี้ช่วยให้คุณ หากต้องการใช้ที่สูงกว่า 5v เพื่อจ่ายไฟให้กับเซอร์โวของคุณ แต่ถ้าแรงดันไฟฟ้าภายนอกของคุณสูงกว่า 5 โวลต์ อย่าเชื่อมต่อเซ็นเซอร์ 5 โวลต์ใดๆ เข้ากับแผงป้องกัน มิฉะนั้น เซ็นเซอร์จะถูกทอดทิ้ง

ขั้นตอนที่ 2: ดาวน์โหลด VarSpeedServo Library

คุณต้องใช้ไลบรารีนี้แทนที่ไลบรารีเซอร์โว Arduino มาตรฐานเนื่องจากช่วยให้คุณสามารถส่งความเร็วเซอร์โวไปยังคำสั่งการเขียนเซอร์โวได้ ห้องสมุดตั้งอยู่ที่นี่:

ห้องสมุด VarSpeedServo

คุณสามารถใช้ปุ่ม zip ดาวน์โหลดไฟล์ zip แล้วติดตั้งด้วย Arduino IDE เมื่อติดตั้งคำสั่งในโปรแกรมของคุณจะมีลักษณะดังนี้: servo.write(100, 20);

พารามิเตอร์แรกคือมุม และพารามิเตอร์ที่สองคือความเร็วของเซอร์โวตั้งแต่ 0 ถึง 255 (ความเร็วเต็มที่)

ขั้นตอนที่ 3: เรียกใช้ Sketch นี้

นี่คือโปรแกรมการแข่งขัน คุณต้องแก้ไขพารามิเตอร์สองสามตัวสำหรับขนาดแขนหุ่นยนต์ของคุณ:

1. BASE_HGT, HUMERUS, ULNA, GRIPPER ความยาวหน่วยเป็นมิลลิเมตร

2. ป้อนหมายเลขพินเซอร์โวของคุณ

3. ป้อนเซอร์โวต่ำสุดและสูงสุดในคำสั่งแนบ

4. จากนั้นลองใช้คำสั่ง set_arm() อย่างง่าย จากนั้นใช้ฟังก์ชัน zero_x(), line() และ circle() สำหรับการทดสอบ ตรวจสอบให้แน่ใจว่าความเร็วเซอร์โวของคุณต่ำในครั้งแรกที่คุณเรียกใช้ฟังก์ชันเหล่านี้ เพื่อป้องกันความเสียหายที่แขนและแขนของคุณเอง

ขอให้โชคดี.

#include VarSpeedServo.h

/* การควบคุมเซอร์โวสำหรับแขน AL5D */

/* ขนาดแขน(mm) */

#define BASE_HGT 90 // ความสูงฐาน

#define HUMERUS 100 // ไหล่ถึงข้อศอก "กระดูก"

#define ULNA 135 // ข้อศอกถึงข้อมือ "กระดูก"

#define GRIPPER 200 //กริปเปอร์ (รวมกลไกการหมุนข้อมือสำหรับงานหนัก) ความยาว"

#define ftl(x) ((x)>=0?(long)((x)+0.5):(long)((x)-0.5)) // การแปลงจากโฟลตเป็นแบบยาว

/* ชื่อ/หมายเลขเซอร์โว *

* ฐานเซอร์โว HS-485HB */

#define BAS_SERVO 4

/* เซอร์โวไหล่ HS-5745-MG */

#define SHL_SERVO 5

/* เซอร์โวข้อศอก HS-5745-MG */

#define ELB_SERVO 6

/* เซอร์โวข้อมือ HS-645MG */

#define WRI_SERVO 7

/* เซอร์โวหมุนข้อมือ HS-485HB */

#define WRO_SERVO 8

/* กริปเปอร์เซอร์โว HS-422 */

#define GRI_SERVO 9

/* การคำนวณล่วงหน้า */

float hum_sq = HUMERUS*HUMERUS;

float uln_sq = ULNA*ULNA;

int servoSPeed = 10;

//ServoShield เซอร์โว; //ServoShield วัตถุ

VarSpeedServo เซอร์โว1, เซอร์โว2, เซอร์โว3, เซอร์โว4, เซอร์โว5, เซอร์โว6;

int loopCounter=0;

int ชีพจรความกว้าง = 6.6;

int microsecondsToDegrees;

การตั้งค่าเป็นโมฆะ ()

{

servo1.attach (BAS_SERVO, 544, 2400);

servo2.attach (SHL_SERVO, 544, 2400);

servo3.attach (ELB_SERVO, 544, 2400);

servo4.attach (WRI_SERVO, 544, 2400);

servo5.attach (WRO_SERVO, 544, 2400);

servo6.attach (GRI_SERVO, 544, 2400);

ล่าช้า (5500);

//servos.start(); //เริ่มเซอร์โวชิลด์

servo_park();

ล่าช้า(4000);

Serial.begin(9600);

Serial.println("เริ่ม");

}

วงเป็นโมฆะ ()

{

loopCounter +=1;

//set_arm(-300, 0, 100, 0, 10); //

// ล่าช้า (7000);

//zero_x();

//ไลน์();

//วงกลม();

ล่าช้า(4000);

ถ้า (loopCounter > 1) {

servo_park();

//set_arm(0, 0, 0, 0, 10); // สวน

ล่าช้า (5000);

ทางออก(0); }//หยุดโปรแกรมชั่วคราว - กดรีเซ็ตเพื่อดำเนินการต่อ

//ออก(0);

}

/* รูทีนการวางตำแหน่งแขนโดยใช้จลนศาสตร์ผกผัน */

/* z คือความสูง y คือระยะห่างจากศูนย์กลางฐาน x คือด้านหนึ่งไปอีกด้าน y, z เป็นค่าบวกได้เท่านั้น */

// เป็นโมฆะ set_arm (uint16_t x, uint16_t y, uint16_t z, uint16_t grip_angle)

เป็นโมฆะ set_arm (float x, float y, float z, float grip_angle_d, int servoSpeed)

{

float grip_angle_r = เรเดียน (grip_angle_d); //มุมกริปเป็นเรเดียนสำหรับใช้ในการคำนวณ

/* มุมฐานและระยะรัศมีจากพิกัด x, y */

float bas_angle_r = atan2(x, y);

float rdist = sqrt((x * x) + (y * y));

/* rdist คือพิกัด y สำหรับแขน */

y = rdist;

/* ค่าออฟเซ็ตของกริปคำนวณจากมุมกริป */

float grip_off_z = (บาป (grip_angle_r)) * GRIPPER;

float grip_off_y = (cos(grip_angle_r)) * GRIPPER;

/* ตำแหน่งข้อมือ */

float wrist_z = (z - grip_off_z) - BASE_HGT;

float wrist_y = y - grip_off_y;

/* ระยะไหล่ถึงข้อมือ (AKA sw) */

float s_w = (wrist_z * wrist_z) + (wrist_y * wrist_y);

ลอย s_w_sqrt = sqrt(s_w);

/* s_w มุมกับพื้น */

float a1 = atan2(wrist_z, wrist_y);

/* s_w มุมถึงกระดูกต้นแขน */

float a2 = acos(((hum_sq - uln_sq) + s_w) / (2 * HUMERUS * s_w_sqrt));

/* มุมไหล่ */

ลอย shl_angle_r = a1 + a2;

float shl_angle_d = องศา (shl_angle_r);

/* มุมศอก */

float elb_angle_r = acos((hum_sq + uln_sq - s_w) / (2 * HUMERUS * ULNA));

float elb_angle_d = องศา (elb_angle_r);

float elb_angle_dn = - (180.0 - elb_angle_d);

/* มุมข้อมือ */

float wri_angle_d = (grip_angle_d - elb_angle_dn) - shl_angle_d;

/* เซอร์โวพัลส์ */

float bas_servopulse = 1500.0 - ((องศา (bas_angle_r)) * pulseWidth);

float shl_servopulse = 1500.0 + ((shl_angle_d - 90.0) * pulseWidth);

float elb_servopulse = 1500.0 - ((elb_angle_d - 90.0) * pulseWidth);

//float wri_servopulse = 1500 + (wri_angle_d * pulseWidth);

//float wri_servopulse = 1500 + (wri_angle_d * pulseWidth);

float wri_servopulse = 1500 - (wri_angle_d * pulseWidth); // อัปเดต 2018/2/11 โดย jimrd - ฉันเปลี่ยนค่าบวกเป็นลบ - ไม่แน่ใจว่าโค้ดนี้ใช้ได้กับทุกคนมาก่อนอย่างไร อาจเป็นไปได้ว่าข้อศอกเซอร์โวถูกติดตั้งโดยหัน 0 องศาแทนที่จะขึ้น

/* ตั้งค่าเซอร์โว */

//servos.setposition(BAS_SERVO, ftl(bas_servopulse));

microsecondsToDegrees = แผนที่ (ftl (bas_servopulse), 544, 2400, 0, 180);

servo1.write (ไมโครวินาทีToDegrees, servoSpeed); // ใช้ฟังก์ชันนี้เพื่อตั้งค่าความเร็วเซอร์โว //

//servos.setposition(SHL_SERVO, ftl(shl_servopulse));

microsecondsToDegrees = แผนที่ (ftl(shl_servopulse), 544, 2400, 0, 180);

servo2.write (ไมโครวินาทีToDegrees, servoSpeed);

//servos.setposition(ELB_SERVO, ftl(elb_servopulse));

microsecondsToDegrees = แผนที่ (ftl(elb_servopulse), 544, 2400, 0, 180);

servo3.write (ไมโครวินาทีToDegrees, servoSpeed);

//servos.setposition(WRI_SERVO, ftl(wri_servopulse));

microsecondsToDegrees = แผนที่ (ftl(wri_servopulse), 544, 2400, 0, 180);

servo4.write (ไมโครวินาทีToDegrees, servoSpeed);

}

/* ย้ายเซอร์โวไปที่ตำแหน่งจอดรถ */

เป็นโมฆะ servo_park()

{

//servos.setposition (BAS_SERVO, 1500);

servo1.write (90, 10);

//servos.setposition (SHL_SERVO, 2100);

servo2.write (90, 10);

//servos.setposition (ELB_SERVO, 2100);

servo3.write (90, 10);

//servos.setposition (WRI_SERVO, 1800);

servo4.write (90, 10);

//servos.setposition (WRO_SERVO, 600);

servo5.write (90, 10);

//servos.setposition (GRI_SERVO, 900);

servo6.write (80, 10);

กลับ;

}

เป็นโมฆะ zero_x()

{

สำหรับ (แกนคู่ = 250.0; yaxis <400.0; yaxis += 1) {

Serial.print(" yaxis=: ");Serial.println(yaxis); อนุกรม.print(" yaxis=: ");

set_arm(0, yaxis, 200.0, 0, 10);

ล่าช้า(10);

}

สำหรับ (แกนคู่ = 400.0; yaxis > 250.0; yaxis -= 1) {

set_arm(0, yaxis, 200.0, 0, 10);

ล่าช้า(10);

}

}

/* ขยับแขนเป็นเส้นตรง */

เส้นเป็นโมฆะ ()

{

สำหรับ (xaxis คู่ = -100.0; xaxis < 100.0; xaxis += 0.5) {

set_arm(xaxis, 250, 120, 0, 10);

ล่าช้า(10);

}

สำหรับ (xaxis ลอย = 100.0; xaxis > -100.0; xaxis -= 0.5) {

set_arm(xaxis, 250, 120, 0, 10);

ล่าช้า(10);

}

}

วงกลมว่างเปล่า ()

{

#กำหนดรัศมี 50.0

// มุมลอย = 0;

zaxis ลอย, yaxis;

สำหรับ (มุมลอย = 0.0; มุม <360.0; มุม += 1.0) {

yaxis = RADIUS * sin(เรเดียน(มุม)) + 300;

zaxis = RADIUS * cos(เรเดียน(มุม)) + 200;

set_arm(0, yaxis, zaxis, 0, 50);

ล่าช้า(10);

}

}

ขั้นตอนที่ 4: ข้อเท็จจริง ปัญหา และสิ่งที่ชอบ…

ข้อเท็จจริง ปัญหา และความชอบ…
ข้อเท็จจริง ปัญหา และความชอบ…

1. เมื่อฉันเรียกใช้รูทีนย่อยของ circle() หุ่นยนต์ของฉันจะเคลื่อนที่เป็นวงรีมากกว่าวงกลม ฉันคิดว่านั่นเป็นเพราะเซอร์โวของฉันไม่ได้รับการปรับเทียบ ฉันทดสอบหนึ่งในนั้นและ 1,500 ไมโครวินาทีไม่เหมือนกับ 90 องศา จะทำงานนี้เพื่อพยายามหาทางแก้ไข อย่าเชื่อว่ามีอะไรผิดปกติกับอัลกอริทึม แต่กับการตั้งค่าของฉัน อัปเดต 2018/2/11 - เพิ่งค้นพบว่าเกิดจากข้อผิดพลาดในรหัสดั้งเดิม ฉันไม่เห็นว่าโปรแกรมของเขาทำงานอย่างไร รหัสคงที่โดยใช้สิ่งนี้: float wri_servopulse = 1500 - (wri_angle_d * pulseWidth); (รหัสเดิมถูกเพิ่ม)

2. ฉันจะหาข้อมูลเพิ่มเติมเกี่ยวกับวิธีการทำงานของฟังก์ชัน set_arm() ได้ที่ไหน: เว็บไซต์ของ Oleg Mazurov อธิบายทุกอย่างหรือให้ลิงก์สำหรับข้อมูลเพิ่มเติม:

3. มีการตรวจสอบเงื่อนไขขอบเขตหรือไม่? ไม่ เมื่อแขนหุ่นยนต์ของฉันถูกส่งผ่านพิกัด xyz ที่ไม่ถูกต้อง มันจะเคลื่อนไหวแบบโค้งอย่างตลกๆ เหมือนกับการยืดเส้นยืดสายของแมว ฉันเชื่อว่าโอเล็กกำลังตรวจสอบโปรแกรมล่าสุดของเขาที่ใช้ USB เพื่อตั้งโปรแกรมการเคลื่อนไหวของแขน ดูวิดีโอของเขาและลิงก์ไปยังรหัสล่าสุดของเขา

4. ต้องล้างรหัสและรหัสไมโครวินาทีก็สามารถทำได้