Operácie

Ovládanie robotického ramienka joystickom

Zo stránky SensorWiki

Záverečný projekt predmetu MIPS / LS2025 - Juraj Krasnovský

(napr. https://www.engineersgarage.com/how-to-use-a-wireless-joystick-to-remotely-control-two-servo-motors/) 

Zadanie

Cieľom projektu bolo ovládať robotické ramienko s tromi servo motormi pomocou pomocou jedného joysticku.

Vývojová doska Arduino Uno.

Literatúra:


Analýza a opis riešenia

Použíté komponenty:

Použitý joystick
Použitý servomotor
Použitý klon Arduino Uno

Joystick funguje ako dva potenciometre, kde zmena polohy mení hodnotu v dvoch smeroch, x a y. Pomocou AD prevodníka som odčítal, že sa tieto hodnoty menia od 0 do 1023, kde 512 je stred. Na ovládanie dvoch servomotorov som použil Timer1. Pre tretí servomotor som musel, kvôli obmedzeniam arduino uno, použiť 8-bitový časovač Timer2,ktorý má síce menší rozsah pre nastavenie PWM ako Timer1, ale pre tento projekt je dostatočný. PWM signál má štandardnú dĺžku impulzov od 1 ms do 2ms. Pre Timer2 som použil prescaler rovný 256, pomocou ktorého som vypočítal dĺžku jedného kroku a potom pomocou nej určil rozsah OCR2:

Rovnaký výpočet som robil aj pre Timer1, menil sa ale iba prescaler (je rovný 8), tak napíšem iba výsledný rozsah OCR1:


Nezabudnite doplniť schému zapojenia! V texte by ste mali opísať základné veci zo zapojenia, samotná schéma nie je dostačujúci opis.

Schéma zapojenia.


Algoritmus a program

Program zabezpečuje ovládanie robotického ramena pomocou odčítaných hodnôt "x" a "y" z AD prevodníka. Rameno sa skladá z troch servomotorov, horný, stredný a dolný. Stredný servomotor je ovládaný neustále podľa hodnoty "y" z joysticku. Horný a dolný servomotor sú ovládané pomocou hodnoty "x".Samotné servá sa pohybujú pomocou premenných dx a dy, ktoré predstavujú "kroky" servomotorčeka. Tieto premenné sa pričítajú/odčítajú v ďaľších premenný posx a posy, podľa hodnôt "x" a "y". Použité funkcie: adc_read – čítanie analógovej hodnoty z joysticku uart_init – inicializácia sériovej komunikácie UART uart_getc – čítanie znaku z UART delay – funkcia na oneskorenie Medzi horným a dolným servomotorom je treba prepínať, keďže chceme v jednom čase ovládať iba jeden z nich. Výber a ovládanie iba jedného je založené na dvoch režimoch, medzi ktorými sa prepína pomocou premennej „rezim“, do ktorej je funkciou getc() načítaný znak. Režim1 (znak = ' + ') zabezpečuje ovládanie spodného motora, tento režim je predvolený. Režim2 (znak =' - ') zabezpečuje ovládanie horného motora. Ak do premennej „znak“ nie je načítané „+“ ani „-“ UART vypíše chybu a ovládanie sa zablokuje.


#define BAUD 9600
#define F_CPU 16000000UL

#include <stdio.h>
#include <avr/io.h>
#include "uart.h"
#include "adc.h"
#include <util/delay.h>

#define set_bit(ADDRESS,BIT) (ADDRESS |= (1<<BIT))
#define clear_bit(ADDRESS,BIT) (ADDRESS &= ~(1<<BIT))
	
FILE mystdout = FDEV_SETUP_STREAM(uart_putc, NULL, _FDEV_SETUP_WRITE);

	
void delay(int delay)
{
  for (int i=1; i<=delay; i++)
  _delay_ms(1);
}	

int main(void)
{
	unsigned int x, y, stred = 512, dolna, horna, posx, posy,dx,dy,xmax,xmin,ymax,ymin;
	char rezim = '+';

	adc_init();
	uart_init();
	stdout = &mystdout;
	
	DDRB|=(1<<PB1);		//dolne servo    
	DDRB |= (1<<PB2);	//horne servo
	DDRB|=(1<<PB3);		//stredne servo
	
	TCNT1 = 0;
	ICR1  = 39999;
	TCCR1A|=(1<<COM1A1)|(1<<COM1B1)|(1<<WGM11);					//Timer1
	TCCR1B|=(1<<WGM12)|(1<<WGM13)|(1<<CS11);  

	TCCR2A |= (1 << COM2A1) | (1 << WGM21) | (1 << WGM20);		//Timer2
	TCCR2B |= (1 << CS21) | (1 << CS22);
	
	OCR1A=3000;		//dolne servo
	OCR1B=3000;		//horne servo
	OCR2A =98;		//stredne servo  
	
	posx = 3000;	//stredna hodnota horneho a dolneho serva
 	posy = 98;		//stredna hodnota stredneho serva

	dx = 40;		//zmena x-ovej hodnoty
	dy = 1;			//zmena y-ovej hodnoty
	
	
	//krajne hodnoty
	xmax = 4000;
	xmin = 2000;
	
	ymax = 125;
	ymin = 62;
	
	//deadzone
	dolna = stred-40;
	horna = stred+40;
	
	while(1)
    {
	  
	x = adc_read(4);
	y = adc_read(5);
	
    //ukladanie zapisaneho znaku
	if (UCSR0A & (1 << RXC0)) {		
            rezim = uart_getc();
	}			
			
	if ( (x < dolna)){
		if(posx >xmin) posx -= dx;
			else posx = xmin; 
	}
	else if(x > horna){
		if(posx <xmax) posx += dx;
			else posx = xmax;
	}
	
	if ( (y < dolna)){
		if(posy >ymin) posy -= dy;
			else posy = ymin; 
	}
	else if(y > horna){
		if(posy <ymax) posy += dy;
			else posy = ymax;
	}
	
	if(rezim == '+'){							//rezim 1
		OCR1A = posx;
		OCR2A = posy;
		printf("rezim:%c\n",rezim);
	}		
	else if(rezim == '-'){						//rezim 2
		OCR1B = posx;
		OCR2A = posy;
		printf("rezim:%c\n",rezim);
		
	}
	else if( (rezim != '+') && (rezim != '-'))	//chybny vstup
		{
			uart_puts("CHYBA zle zvoleny rezim\n");
		}				
		
	delay(5);
		
    }
}
#include <avr/io.h>

void uart_init( void );
     
void uart_putc( char c );
void uart_puts( const char *s );

char uart_getc( void );
#define BAUD 9600
#define F_CPU 16000000UL
#define BAUDRATE       9600
#define BAUD_PRESCALE  103  // vzor?ek z datasheetu

#include <avr/io.h>
#include <util/setbaud.h>

void uart_init( void ) 
{
    UBRR0H = UBRRH_VALUE;
    UBRR0L = UBRRL_VALUE;

#if USE_2X
    UCSR0A |= _BV(U2X0);
#else
    UCSR0A &= ~(_BV(U2X0));
#endif

    UCSR0C = _BV(UCSZ01) | _BV(UCSZ00); /* 8-bit data */
    UCSR0B = _BV(RXEN0) | _BV(TXEN0);   /* Enable RX and TX */
}


void uart_putc(char c) 
{
   if (c == '\n') 
    {
       uart_putc('\r');
    }
   loop_until_bit_is_set(UCSR0A, UDRE0); /* Wait until data register empty. */
   UDR0 = c;
}


void uart_puts(const char *s)
{
	while(*s!='\0')
	{
		uart_putc(*s++);
	}
}

char uart_getc(void) {
    loop_until_bit_is_set(UCSR0A, RXC0); /* Wait until data exists. */
    return UDR0;
}
#include <avr/io.h>

void adc_init(void);                                   // A/D converter initialization

unsigned int adc_read(char a_pin);
#include <avr/io.h>

void adc_init(void)
{
    ADMUX = (1<<REFS0);         // Vref bude Avcc
    ADCSRA = (1<<ADEN)          // AD Enable - "zapnutie" ADC
            |(1<<ADPS2)|(1<<ADPS1)|(1<<ADPS0);          // nastavenie preddelica
}

unsigned int adc_read(char channel)
{
    channel &= 0x0F;
    ADMUX = (ADMUX & 0xF0)|channel;
    ADCSRA |= (1<<ADSC);        // spustenie prevodu
    while(ADCSRA & (1<<ADSC))
    { /* cakaj tu */ }          // pockam na dokoncenie prevodu
    return (ADC);
}

Pridajte sem aj zbalený kompletný projekt, napríklad takto (použite jednoznačné pomenovanie, nemôžeme mať na serveri 10x zdrojaky.zip:

Zdrojový kód: zdrojaky.zip

Overenie

Funkciu som overil zapojením a následným vyskúšaním ovládania.

Aplikácia.

Video:



Kľúčové slová 'Category', ktoré sú na konci stránky nemeňte.