- Labornetzteil AliExpress         
Ergebnis 1 bis 10 von 10

Thema: Probleme bei Programmierung der Snake Vision für ASURO

Hybrid-Darstellung

Vorheriger Beitrag Vorheriger Beitrag   Nächster Beitrag Nächster Beitrag
  1. #1
    Neuer Benutzer Öfters hier
    Registriert seit
    10.05.2011
    Beiträge
    7

    Probleme bei Programmierung der Snake Vision für ASURO

    Ich versuche schon seit einiger Zeit meinen ASURO, den ich mit der Snake Vision erweitert habe, dazu zu bringen, dass er auf die wärmste Quelle, die er registriert zufährt, und wenn er an sie stößt ein stückweit rückwärts fahren.Danach sollte er nichts mehr tun. So leicht es sich auch anhört, bin ich mit meinen sehr beschränkten Programmierfähigkeiten nicht weit gekommen.
    Code:
    #include "asuro.h"
    
    void ThermalData(unsigned int *data)
    {
    	
    	ADMUX = (1 << REFS0) | (1 << REFS1) | IR_LEFT;	
    		ADCSRA |= (1 << ADSC);    
    	while (!(ADCSRA & (1 << ADIF)));	                
    		ADCSRA |= (1 << ADIF);			        
    	
    	data[0] = ADCL + (ADCH << 8);
    
    	
    	ADMUX = (1 << REFS0) | (1 << REFS1) | IR_RIGHT;
    		ADCSRA |= (1 << ADSC);
    		while (!(ADCSRA & (1 << ADIF)));	
    		ADCSRA |= (1 << ADIF);
    	data[1] = ADCL + (ADCH << 8);
    }
    
    
    int main(void)
    {
    	
        	unsigned int tdata[2];
    		signed int diff,sum;
        	Init();
    	
    	
    	while(1)
        	{
    	
            	ThermalData(tdata);
    	
    		sum=tdata[0]+tdata[1];
    		StatusLED(RED);
    
    		if(PollSwitch()==0){
             MotorDir(FWD,FWD);
    
    	
    			diff=((signed)tdata[0]-(signed)tdata[1])*32/sum;
    			if (diff>4) {
    				
    				BackLED(ON,OFF); 
    				MotorSpeed(0,140);
    			}
    			else if (diff<-4) {
    			
    				BackLED(OFF,ON); 
    				MotorSpeed(140, 0);
    			}
    			else {
    			
    				BackLED(OFF,OFF); 
    				MotorSpeed(140, 140);
    			}
    	}
    
    		else {
    			
    			StatusLED(GREEN);
    			BackLED(OFF,OFF);
    			MotorDir(RWD,RWD);
    			MotorSpeed(140,140);
    			Sleep(2000);
    
           
    		}
        	}     
    	return 0;
    }
    Mit diesem Programm fährt mein ASURO lediglich rechtsrum im Kreis und fährt nur solange rückwärts wie ein Taster gedrückt wird, obwohl er nach Tasterdrück eine gewisse zeitlang rückwärtsfahren soll.

    Was muss ich verändern, damit das Programm funktioniert wie ich es will??? Oder ist schon der Ansatz an sich falsch???

    Ich freu mich auf jede nur erdenkliche Rückmeldung, da ich selbst langsam nichts mehr verstehe.....

    MfG nooby21

  2. #2
    Erfahrener Benutzer Robotik Einstein Avatar von SprinterSB
    Registriert seit
    09.06.2005
    Ort
    An der Saar
    Beiträge
    2.802
    Zitat Zitat von nooby21 Beitrag anzeigen
    Code:
    data[...] = ADCL + (ADCH << 8);
    Holla, das ist ja wie Russisch Roulette. Eine 50%-Wahrscheinlichkeit daß die Zugriffer wie von der Hardware vorgesehen passieren.

    Warum nicht?
    Code:
    data[...] = ADC;
    Disclaimer: none. Sue me.

  3. #3
    Benutzer Stammmitglied Avatar von KR-500
    Registriert seit
    26.12.2007
    Alter
    29
    Beiträge
    91
    Hi,

    also als erstes würde ich mal empfehlen die Daten die gemessen werden über UART ausgeben zu lassen um einfach mal zu gucken was da los ist. Anstatt dem hier:
    Code:
    data[...] = ADCL + (ADCH << 8);
    könntest du auch:
    Code:
    data[...] = ADCW;
    schreiben. Ein weiterer Pukt an dem du ansetzen könntest wäre die Referenzspannung. Ich selber habe keine ASURO, habe mir aber mal das Snake Vision gekauft. Ich habe zum auswerten die interne Referenzspannung von 5 Volt benutzt. Ich habe dann übrigens schnell festgestellt das die Sensoren nur auf eine Entfernung von 30cm zu gebruachen sind. Wenn mein Ziel(Teelicht) weiter enfernt war dann haben die Sensoren nichts mehr feststellen können. Um auf die wärmste Quelle zu zufahren habe ich auch einen P-Regler verwndet das hat sehr gut geklappt und sah etwa so aus:
    Code:
           int ocr1a=0,ocr1b=0,dif=0,regler_kp=25,regler=0;
           while((adc1<(pir+50))&&(adc2<(pir+50))){
                adc1 = read_adc3();
                adc2 = read_adc4();
                dif = adc1 - adc2;
    
                regler_kp = read_adc2()-128;
    
                regler = (dif * regler_kp)/10;            
             
                speed = read_adc1();
    
                ocr1a = speed - regler;
                ocr1b = speed + regler;    
    
                if(ocr1a<0){ ocr1a = 0;}
                if(ocr1b<0){ ocr1b = 0;}
                if(ocr1a>255){ ocr1a = 255;}
                if(ocr1b>255){ ocr1b = 255;}
                OCR1A = ocr1a;
                OCR1B = ocr1b;
            }
    An ADC1 und ADC2 habe ich noch Potentiometer um eine gewisse Grundgeschwindigkeit und die Regler_kp einzustellen. Und in der while-Schleife bleibt er solange bis er nah genug am Teelicht ist.

    KR-500

  4. #4
    Neuer Benutzer Öfters hier
    Registriert seit
    10.05.2011
    Beiträge
    7
    Ich hab nun einfach das Standartprogramm übernommen und eine weitere If-Schleife eingefügt, die den ASURO zum anhalten bringen soll, wenn ein Taster gedrückt wird. Der ASURO fährt jetzt auf meine Wärmequelle (eine Teelicht) zu, zwar etwas ruckartig, aber es funktioniert. Das Anhalten bei Tasterdruck geht aber weiterhin nicht...

    Code:
    #include "asuro.h"
    
    // Schwellen für die intensitätsabhängige Geschwindigkeit
    
    #define THRESH1 20
    
    
    void ThermalData(unsigned int *data)
    {
    	// ThermalData() funktioniert genauso, wie LineData(), ...
    	// ... nur dass dabei die interne Spannungsreferenz als ...
    	// ... AD-Wandlerreferenz verwendet wird.
    	
    	// Prozessorinterne Referenz verwenden
    	// Multiplexer auf linken Sensor schalten
    	ADMUX = (1 << REFS0) | (1 << REFS1) | IR_LEFT;	
    	// Wandlung starten
    	ADCSRA |= (1 << ADSC);
    	// Warten, bis Wandlung beendet			        
    	while (!(ADCSRA & (1 << ADIF)));	                
    	// ADCIF zurücksetzen
    	ADCSRA |= (1 << ADIF);			        
    	// ADC-Wert auslesen
    	data[0] = ADCL + (ADCH << 8);
    
    	// Prozessorinterne Referenz verwenden
    	// Multiplexer auf rechten Sensor schalten
    	ADMUX = (1 << REFS0) | (1 << REFS1) | IR_RIGHT;
    	// Wandlung starten
    	ADCSRA |= (1 << ADSC);
    	// Warten, bis Wandlung beendet		
    	while (!(ADCSRA & (1 << ADIF)));	
    	// ADCIF zurücksetzen
    	ADCSRA |= (1 << ADIF);
    	// ADC-Wert auslesen
    	data[1] = ADCL + (ADCH << 8);
    }
    
    
    int main(void)
    {
    	// Speicher für die Messwerte bereitstellen
        	unsigned int tdata[2];
    	unsigned int speed;
    	signed int diff,sum;
        	Init();
    	// Motoren immer auf vorwärts
    	MotorDir(FWD,FWD);
    	
    	while(1)
        	{
    		// Messwerte einlesen
            	ThermalData(tdata);
    		// Die Summe der Werte wird für die Berechnung der ...
    		// ... Geschwindigkeit genutzt
    		sum=tdata[0]+tdata[1];
    		StatusLED(RED);
    	
     		// Signal stark genug?
    		if (sum>THRESH1) {
    			
    			speed=140;
    			
    						
    			// Richtung bestimmen
    			diff=((signed)tdata[0]-(signed)tdata[1])*32/sum;
    			if (diff>4) {
    				// Links deutlich wärmer? Nach links fahren!
    				BackLED(ON,OFF); 
    				MotorSpeed(0,speed);
    			}
    			else if (diff<-4) {
    				// Rechts deutlich wärmer? Nach rechts fahren!
    				BackLED(OFF,ON); 
    				MotorSpeed(speed, 0);
    			}
    			else {
    				// Sonst geradeaus!
    				BackLED(OFF,OFF); 
    				MotorSpeed(speed, speed);
    			}
    			if(PollSwitch()){
    			MotorSpeed(0,0);
    			StatusLED(GREEN);}
    		
    		}
    
    		else {
    			// Keine Wärmequelle in Sicht? Stehen bleiben!
    			StatusLED(OFF);
    			BackLED(OFF,OFF);
    		
    		
           
    		}
        	}     
    	return 0;
    }

  5. #5
    Neuer Benutzer Öfters hier
    Registriert seit
    10.05.2011
    Beiträge
    7
    hat wirklcih keiner ne ahnung was ich tun muss? wär echt net wenn mal jemand über mein problem drüber schaun würd...

  6. #6
    Neuer Benutzer Öfters hier
    Registriert seit
    10.05.2011
    Beiträge
    7
    Ich konnte mein Problem weitgehenst beheben. Allerdings kenne ich keinen Befehl, der den ASURO dazu bringt, nachdem er rückwärts gefahren ist, das gesamte Programm zu unterbrechen und einfach nichts zu tun...

    Code:
    #include "asuro.h"
    
    // Schwellen für die intensitätsabhängige Geschwindigkeit
    
    #define THRESH1 20
    #define THRESH2 80
    #define THRESH3 200
    
    
    void ThermalData(unsigned int *data)
    {
    	// ThermalData() funktioniert genauso, wie LineData(), ...
    	// ... nur dass dabei die interne Spannungsreferenz als ...
    	// ... AD-Wandlerreferenz verwendet wird.
    	
    	// Prozessorinterne Referenz verwenden
    	// Multiplexer auf linken Sensor schalten
    	ADMUX = (1 << REFS0) | (1 << REFS1) | IR_LEFT;	
    	// Wandlung starten
    	ADCSRA |= (1 << ADSC);
    	// Warten, bis Wandlung beendet			        
    	while (!(ADCSRA & (1 << ADIF)));	                
    	// ADCIF zurücksetzen
    	ADCSRA |= (1 << ADIF);			        
    	// ADC-Wert auslesen
    	data[0] = ADCL + (ADCH << 8);
    
    	// Prozessorinterne Referenz verwenden
    	// Multiplexer auf rechten Sensor schalten
    	ADMUX = (1 << REFS0) | (1 << REFS1) | IR_RIGHT;
    	// Wandlung starten
    	ADCSRA |= (1 << ADSC);
    	// Warten, bis Wandlung beendet		
    	while (!(ADCSRA & (1 << ADIF)));	
    	// ADCIF zurücksetzen
    	ADCSRA |= (1 << ADIF);
    	// ADC-Wert auslesen
    	data[1] = ADCL + (ADCH << 8);
    }
    
    
    int main(void)
    {
    	// Speicher für die Messwerte bereitstellen
        	unsigned int tdata[2];
    	unsigned int speed;
    	signed int diff,sum;
        	Init();
    	// Motoren immer auf vorwärts
    	MotorDir(FWD,FWD);
    	while(1)
        	{
    		// Messwerte einlesen
            	ThermalData(tdata);
    		// Die Summe der Werte wird für die Berechnung der ...
    		// ... Geschwindigkeit genutzt
    		sum=tdata[0]+tdata[1];
     		// Signal stark genug?
    		if (sum>THRESH1) {
    			
    			speed=140;
    			// Ziemlich stark?
    			if (sum>THRESH2) {
    				
    				speed=140;
    			}
    			// Sehr stark?
    			if (sum>THRESH3) {
    				StatusLED(OFF);
    				speed=140;
    			}
    			// Richtung bestimmen
    			diff=((signed)tdata[0]-(signed)tdata[1])*32/sum;
    			if (diff>4) {
    				// Links deutlich wärmer? Nach links fahren!
    				BackLED(ON,OFF); 
    				MotorSpeed(0,speed);
    			}
    			else if (diff<-4) {
    				// Rechts deutlich wärmer? Nach rechts fahren!
    				BackLED(OFF,ON); 
    				MotorSpeed(speed, 0);
    			}
    			else {
    				// Sonst geradeaus!
    				BackLED(OFF,OFF); 
    				MotorSpeed(speed, speed);
    
    				if(PollSwitch()){
    				MotorSpeed(0,0);
    				Msleep(900);
    				MotorDir(RWD,RWD);
    				MotorSpeed(140,140);
    				Msleep(900);
    				StatusLED(GREEN);}
    			}
    		}
    		//else if(PollSwitch()){
    		//MotorSpeed(0,0);
    		//StatusLED(GREEN);}
    		
    		else {
    			// Keine Wärmequelle in Sicht? Stehen bleiben!
    			StatusLED(OFF);
    			BackLED(OFF,OFF);
    			MotorSpeed(0,0);
    
          
    
    		}
        	}     
    	return 0;
    	}

  7. #7
    Neuer Benutzer Öfters hier
    Registriert seit
    10.05.2011
    Beiträge
    7
    @SprinterSB

    Leider hat sich weder mit ADC noch mit ADCW etwas verändert. Außerdem bin ich mir recht sicher, dass das Einlesen der Messwerte funktioniert, da ich diesen Teil des Programms aus dem Standartprogramm aus "Mehr Spaß mit ASURO 2" übernommen habe.

    Hier mal das Standartprogramm:
    Code:
    #include "asuro.h"
    
    // Schwellen für die intensitätsabhängige Geschwindigkeit
    
    #define THRESH1 20
    #define THRESH2 80
    #define THRESH3 200
    
    
    void ThermalData(unsigned int *data)
    {
    	// ThermalData() funktioniert genauso, wie LineData(), ...
    	// ... nur dass dabei die interne Spannungsreferenz als ...
    	// ... AD-Wandlerreferenz verwendet wird.
    	
    	// Prozessorinterne Referenz verwenden
    	// Multiplexer auf linken Sensor schalten
    	ADMUX = (1 << REFS0) | (1 << REFS1) | IR_LEFT;	
    	// Wandlung starten
    	ADCSRA |= (1 << ADSC);
    	// Warten, bis Wandlung beendet			        
    	while (!(ADCSRA & (1 << ADIF)));	                
    	// ADCIF zurücksetzen
    	ADCSRA |= (1 << ADIF);			        
    	// ADC-Wert auslesen
    	data[0] = ADCL + (ADCH << 8);
    
    	// Prozessorinterne Referenz verwenden
    	// Multiplexer auf rechten Sensor schalten
    	ADMUX = (1 << REFS0) | (1 << REFS1) | IR_RIGHT;
    	// Wandlung starten
    	ADCSRA |= (1 << ADSC);
    	// Warten, bis Wandlung beendet		
    	while (!(ADCSRA & (1 << ADIF)));	
    	// ADCIF zurücksetzen
    	ADCSRA |= (1 << ADIF);
    	// ADC-Wert auslesen
    	data[1] = ADCL + (ADCH << 8);
    }
    
    
    int main(void)
    {
    	// Speicher für die Messwerte bereitstellen
        	unsigned int tdata[2];
    	unsigned int speed;
    	signed int diff,sum;
        	Init();
    	// Motoren immer auf vorwärts
    	MotorDir(FWD,FWD);
    	while(1)
        	{
    		// Messwerte einlesen
            	ThermalData(tdata);
    		// Die Summe der Werte wird für die Berechnung der ...
    		// ... Geschwindigkeit genutzt
    		sum=tdata[0]+tdata[1];
     		// Signal stark genug?
    		if (sum>THRESH1) {
    			StatusLED(GREEN);
    			speed=140;
    			// Ziemlich stark?
    			if (sum>THRESH2) {
    				StatusLED(YELLOW); 
    				speed=200;
    			}
    			// Sehr stark?
    			if (sum>THRESH3) {
    				StatusLED(RED); 
    				speed=255;
    			}
    			// Richtung bestimmen
    			diff=((signed)tdata[0]-(signed)tdata[1])*32/sum;
    			if (diff>4) {
    				// Links deutlich wärmer? Nach links fahren!
    				BackLED(ON,OFF); 
    				MotorSpeed(0,speed);
    			}
    			else if (diff<-4) {
    				// Rechts deutlich wärmer? Nach rechts fahren!
    				BackLED(OFF,ON); 
    				MotorSpeed(speed, 0);
    			}
    			else {
    				// Sonst geradeaus!
    				BackLED(OFF,OFF); 
    				MotorSpeed(speed, speed);
    			}
    		}
    		else {
    			// Keine Wärmequelle in Sicht? Stehen bleiben!
    			StatusLED(OFF);
    			BackLED(OFF,OFF);
    			MotorSpeed(0,0);
    
    		}
        	}     
    	return 0;
    }
    ich glaube nun ist auch zu erkennen, dass ich dieses Programm einfach modifizieren wollte...
    Mein Grundgedanke ist, dass ich if(sum>THRESH1) einfach durch if(PollSwitch()==0) ersetze, so dass ASURO anhält sobald ein Taster gedrückt wird.
    Doch nicht mal so was funktioniert -.-

  8. #8
    Neuer Benutzer Öfters hier
    Registriert seit
    10.05.2011
    Beiträge
    7
    @KR-500

    Dein Programm hört sich sehr gut an. Allerdings glaub ich, dass so etwas für mein Verständnis um einiges zu hoch ist. Außerdem bekomm ich viele Promblemmeldungen bei deinem Programm ,da Befehle wie adc1 oder pir nicht definiert sind.

    Ich vermute, dass mein Problem in der while-Schleife sitzt.

Ähnliche Themen

  1. Projekt: RP6 mit Snake Vision
    Von Thund3r im Forum Robby RP6
    Antworten: 10
    Letzter Beitrag: 21.09.2011, 19:58
  2. Snake Vision
    Von Mangoon im Forum C - Programmierung (GCC u.a.)
    Antworten: 7
    Letzter Beitrag: 12.05.2010, 17:51
  3. Taster auf ASURO Snake Vision-Platine ansprechen
    Von Zehplusplus im Forum Asuro
    Antworten: 3
    Letzter Beitrag: 04.04.2010, 13:53
  4. Snake Vision will nicht.
    Von Obi-Wan1234 im Forum Asuro
    Antworten: 2
    Letzter Beitrag: 18.06.2008, 13:19
  5. Asuro "Snake Vision"
    Von Alina89 im Forum Asuro
    Antworten: 18
    Letzter Beitrag: 21.01.2008, 20:30

Stichworte

Berechtigungen

  • Neue Themen erstellen: Nein
  • Themen beantworten: Nein
  • Anhänge hochladen: Nein
  • Beiträge bearbeiten: Nein
  •  

LiFePO4 Speicher Test