Сервопривод вместо ШД.
- Войдите на сайт для отправки комментариев
Ср, 01/06/2016 - 17:36
Здравствуйте. Помогите пожалуйста. Сам совсем ни-ни. Есть скетч. Управлениие ЧПУ станком. Помогите пожалуйста воткнуть управление сервоприводом вместо оси Z. Заранее спасибо
int motorPins[3][4] = {{2, 3, 4, 5},{6, 7, 8, 9},{10, 11, 12, 13}};
int count;
int count2[3] = {0,0,0};
int val = 0;
int rot=0;
int incomingByte = 0;
int sign=1;
long delayTime;
int s_p=40;
//Процедура настройки прошивки
void setup() {
int i;
Serial.begin(9600);
pinMode(A2, OUTPUT);
for (i=0; i<3; i++) {
for (count = 0; count < 4; count++) {
pinMode(motorPins[i][count], OUTPUT); //установка режима работы цифровых pin'ов Ардуино
}
}
delayTime=2000; //задержка между шагами 3 мс (2 мс - время реакции ШД на подачу напряжения)
}
//Поворот двигателя с номерм sm на один шаг вперёд в направлении st (-1 или 1)
void moveDir(int sm, int st) {
if ((st==1 ) && (count2[sm]) == 4) count2[sm] = 0;
if ((st==-1) && (count2[sm]) == 1) count2[sm] = 5;
count2[sm]+=st;
switch (count2[sm]) {
case 1:
digitalWrite(motorPins[sm][0],HIGH);
digitalWrite(motorPins[sm][1],LOW);
digitalWrite(motorPins[sm][2],LOW);
digitalWrite(motorPins[sm][3],HIGH);
break;
case 2:
digitalWrite(motorPins[sm][0],LOW);
digitalWrite(motorPins[sm][1],HIGH);
digitalWrite(motorPins[sm][2],LOW);
digitalWrite(motorPins[sm][3],HIGH);
break;
case 3:
digitalWrite(motorPins[sm][0],LOW);
digitalWrite(motorPins[sm][1],HIGH);
digitalWrite(motorPins[sm][2],HIGH);
digitalWrite(motorPins[sm][3],LOW);
break;
case 4:
digitalWrite(motorPins[sm][0],HIGH);
digitalWrite(motorPins[sm][1],LOW);
digitalWrite(motorPins[sm][2],HIGH);
digitalWrite(motorPins[sm][3],LOW);
break;
}
}
//Поворот двигателя с номерм sm на один шаг вперёд
void moveForward(int sm) {
moveDir(sm,1);
}
//Поворот двигателя с номерм sm на один шаг назад
void moveBackward(int sm) {
moveDir(sm,-1);
}
//Задержка в микосекундах
void delayMicros(long wt){
unsigned long mls;
unsigned int mks;
mls=(unsigned long)(wt / 1000);
mks=(unsigned int)(wt % 1000);
if (mls>0) delay(mls);
if (mks>0) delayMicroseconds(mks);
}
//Одновременный поворот двигателей 0, 1, 2 на x, y, z шагов соответственно
void MoveSM(long x, long y, long z) {
long c[3], c2[3];
double c1[3], d[3];
long m, i;
boolean flg;
long ms;
long d1, d2, d3, d4, t, dt;
c[0] = x;
c[1] = y;
c[2] = z;
m = 1;
for (i=0; i<2; i++) {
if (m < abs(c[i])) m = abs(c[i]);
}
for (i=0; i<2; i++) {
c1[i] = 0;
d[i] = 1.0 * c[i] / m;
c2[i] = 0;
}
flg = false;
for (i=0; i<2; i++) {
if (abs(c1[i]) < abs(c[i])) flg=true;
}
t=m;
if (z>0 && t>0) {
if (z>=delayTime) {
dt=z; d1=s_p; d2=1;
d3=0;
if (d1==d2) d4=z/2; else d4=z;
} else {
dt=delayTime;
d3=dt*s_p*(dt-z)/dt/2;
d2=d3/dt;
d1=s_p-d2;
d3=d3-d2*dt;
d2=d2+1;
if (d1==d2) d4=dt/2-d3; else d4=dt-d3;
}
} else {
dt=delayTime;
d1=-1; d2=-1; d3=0; d4=0;
}
while (flg) {
flg=false;
for (i=0; i<2; i++) {
if (abs(c1[i]) < abs(c[i]))
c1[i] += d[i];
if (abs(c1[i]) - abs(c2[i]) >= 0.5) {
if (c[i]>0) {
c2[i]++;
moveForward(i);
} else {
c2[i]--;
moveBackward(i);
}
}
if (abs(c1[i]) < abs(c[i])) flg=true;
}
if (z>0) {
if (t==d1) {
delayMicros(d3);
digitalWrite(A2, HIGH);
delayMicros(d4);
}
if (t==d2) {
delayMicros(d4);
digitalWrite(A2, LOW);
delayMicros(d3);
}
if (t!=d1 && t!=d2) {
delayMicros(dt);
}
} else {
delayMicros(dt);
}
t--;
}
}
//Основной цикл
void loop() {
if (Serial.available() > 0) { //Пришла команда
long c[5]={0,0,0,0,0};
int i;
sign=1;
i=0;
incomingByte = Serial.read();
while (incomingByte!=';') { //Читаем входящую строку, признак конца строки знак "точка с запятой"
if (c[i]==0) {
if (incomingByte=='-')
sign=-1;
}
if (incomingByte==',') {
c[i]*=sign;
sign=1;
i++;
} else if (incomingByte>='0' && incomingByte<='9') {
c[i]=c[i]*10+incomingByte-'0';
}
while (Serial.available() == 0) {
delayMicroseconds(1);
}
incomingByte = Serial.read();
}
c[i]*=sign;
if (c[3]>0) s_p=c[3]; //количество шагов на 1 пиксель
if (c[4]>0) delayTime=c[4]; //количество шагов на 1 пиксель
MoveSM(c[0],c[1],c[2]); //Вращаем двигатели на заданное число шагов
Serial.println("OK"); //Отправляем компьютеру сообщение "OK", значит можно высылать новую команду
}
else
delayMicroseconds(1); //Если ничего не пришло, ждём 1 микросекунду
}
Я бы реализовал это так, через подключение библиотек Stepper и Servo. Комментарии сделал для себя, вдруг и Вам пригодятся. Пусть и поздно, но всё же кому-то это может поможет. Тоже начал разбираться в этой теме и наткнулся на этот вопрос, оставленный без ответа.
Не понимаю, в чем тут можно "разбираться" !? Все более менее стандартные контролеры шаговых двигателей имеют всего 2 контакта, направление и шаг. Совершенно так же работает и сервопривод, для контролера это идентичные устройства. Просто берете шаговик с контролером шаговика , выбрасываете его, а на его место ставите контролер сервопривода с сервомотором.
В данном примере было реализовано подлкючение шаговых двигателей из CD-rom'ов через микросхему L293d. Ну или можно через L298N (вроде так маркируется). Собирал из того, что было. Соответственно у такого подключения задействованы 4 пина. Можно конечно сократить и до двух, но через инвертирование вторых выводов управления, а они уже не направление и шаг.
Нужно просто изменить подпрограмму шага в библиотеке (void Stepper::stepMotor(int thisStep)) или использовать другую библиотеку.
Спасибо, буду теперь знать)
Эх....
Вот если использовать эту процедуру, то для двухпинового подключения на пин 1 посадить DIR, на пин 2 - STEP, сам не проверял, но имхо должно работать. Четырехпиновое работать не будет вообще. Как это использовать уже подумайте сами.