mirror of
				https://github.com/RobotechLille/cdf2018-principal
				synced 2025-10-24 17:53:31 +02:00 
			
		
		
		
	Better IHM
This commit is contained in:
		
							parent
							
								
									553c550ac7
								
							
						
					
					
						commit
						760b950e83
					
				
					 17 changed files with 244 additions and 211 deletions
				
			
		|  | @ -10,6 +10,9 @@ LDFLAGS_CUSTOM += -lpthread -lwiringPi | ||||||
| CFLAGS_CUSTOM += -g | CFLAGS_CUSTOM += -g | ||||||
| ## Générateurs de drapeaux pour les bibliothèques
 | ## Générateurs de drapeaux pour les bibliothèques
 | ||||||
| PKG_CONFIG=pkg-config | PKG_CONFIG=pkg-config | ||||||
|  | ## Nom des objets communs
 | ||||||
|  | OBJS=CF debug i2c ihm lcd movement parcours points position | ||||||
|  | OBJS_O=$(addprefix obj/,$(addsuffix .o,$(OBJS))) | ||||||
| 
 | 
 | ||||||
| # VARIABLES AUTOMATIQUES
 | # VARIABLES AUTOMATIQUES
 | ||||||
| ifdef LIBS | ifdef LIBS | ||||||
|  | @ -21,25 +24,20 @@ endif | ||||||
| CFLAGS += -Wall -Wextra -pedantic -g -DDEBUG | CFLAGS += -Wall -Wextra -pedantic -g -DDEBUG | ||||||
| # buildroot se charge de remplacer ces flags avec des optimisations
 | # buildroot se charge de remplacer ces flags avec des optimisations
 | ||||||
| 
 | 
 | ||||||
| # RÈGLES AUTOMATIQUES DE COMPILATION
 | # RÈGLES DE COMPILATION
 | ||||||
|  | 
 | ||||||
|  | # Règle éxecutée par défaut (quand on fait juste `make`)
 | ||||||
|  | default: bin/premier bin/local $(subst src,bin,$(subst .c,,$(wildcard src/test*.c))) | ||||||
| 
 | 
 | ||||||
| # Génération des fichiers éxecutables
 | # Génération des fichiers éxecutables
 | ||||||
| bin/%: obj/%.o | bin/%: obj/%.o $(OBJS_O) | ||||||
| 	$(CC) $(LDFLAGS) $(LDFLAGS_CUSTOM) $^ -o $@ | 	$(CC) $(LDFLAGS) $(LDFLAGS_CUSTOM) $^ -o $@ | ||||||
| 	$(OBJCOPY) --only-keep-debug $@ $@.debug | 	$(OBJCOPY) --only-keep-debug $@ $@.debug | ||||||
| 	$(STRIP) --strip-debug --strip-unneeded $@ | 	$(STRIP) --strip-debug --strip-unneeded $@ | ||||||
| 
 | 
 | ||||||
| # RÈGLES DE COMPILATION
 |  | ||||||
| 
 |  | ||||||
| # Règle éxecutée par défaut (quand on fait juste `make`)
 |  | ||||||
| default: bin/testpin bin/premier bin/local bin/testI2c |  | ||||||
| 
 |  | ||||||
| # Binaires (dont il faut spécifier les objets explicitement)
 | # Binaires (dont il faut spécifier les objets explicitement)
 | ||||||
| OBJS=CF movement debug position ihm lcd i2c points parcours | bin/premier: obj/premier.o $(OBJS_O) | ||||||
| bin/premier: $(addprefix obj/,$(addsuffix .o,$(OBJS))) | bin/test%: obj/test%.o $(OBJS_O) | ||||||
| bin/testPin: obj/testPin.o |  | ||||||
| # TODO ↑ Enlever (remplacé par IHM)
 |  | ||||||
| bin/testI2c: obj/testI2c.o obj/i2c.o obj/srf08.o obj/lcd.o |  | ||||||
| 
 | 
 | ||||||
| # Programme de test sur PC, n'embarquant pas wiringPi
 | # Programme de test sur PC, n'embarquant pas wiringPi
 | ||||||
| bin/local: obj/local.o obj/debug.o | bin/local: obj/local.o obj/debug.o | ||||||
|  |  | ||||||
							
								
								
									
										12
									
								
								chef/run.sh
									
										
									
									
									
								
							
							
						
						
									
										12
									
								
								chef/run.sh
									
										
									
									
									
								
							|  | @ -1,12 +0,0 @@ | ||||||
| #!/bin/sh |  | ||||||
| 
 |  | ||||||
| cd "$( dirname "${BASH_SOURCE[0]}" )" |  | ||||||
| EXEC=bin/premier |  | ||||||
| LOGFILE="${2:=run.log}" |  | ||||||
| 
 |  | ||||||
| # Logging |  | ||||||
| "$EXEC" 2>&1 | while read line; do |  | ||||||
|     echo "$(cat /proc/uptime | cut -d ' ' -f 1) $line" >> "$LOGFILE" |  | ||||||
| done |  | ||||||
| 
 |  | ||||||
| sh lcdOff.sh |  | ||||||
|  | @ -114,7 +114,7 @@ void configureCF() | ||||||
|     pthread_mutex_init(&sSendCF, NULL); |     pthread_mutex_init(&sSendCF, NULL); | ||||||
|     pthread_create(&tReaderAC, NULL, TaskReaderAC, NULL); |     pthread_create(&tReaderAC, NULL, TaskReaderAC, NULL); | ||||||
| 
 | 
 | ||||||
|     printf("Attente de réponse de l'Fpga... "); |     printf("Attente de réponse du Fpga... "); | ||||||
|     fflush(stdout); |     fflush(stdout); | ||||||
|     struct timespec tim; |     struct timespec tim; | ||||||
|     tim.tv_sec = 0; |     tim.tv_sec = 0; | ||||||
|  |  | ||||||
|  | @ -20,7 +20,7 @@ void* TaskDebug(void* pdata) | ||||||
|     (void)pdata; |     (void)pdata; | ||||||
| 
 | 
 | ||||||
|     clock_t debugStart; |     clock_t debugStart; | ||||||
|     debugStart = clock(); |     debugStart = clock(); // TODO struct timespec
 | ||||||
| 
 | 
 | ||||||
|     struct timespec tim; // 100 ms
 |     struct timespec tim; // 100 ms
 | ||||||
|     tim.tv_sec = 0; |     tim.tv_sec = 0; | ||||||
|  |  | ||||||
							
								
								
									
										109
									
								
								chef/src/ihm.c
									
										
									
									
									
								
							
							
						
						
									
										109
									
								
								chef/src/ihm.c
									
										
									
									
									
								
							|  | @ -1,6 +1,5 @@ | ||||||
| #include <pthread.h> | #include <pthread.h> | ||||||
| #include <signal.h> | #include <signal.h> | ||||||
| #include <stdbool.h> |  | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
| #include <time.h> | #include <time.h> | ||||||
| #include <wiringPi.h> | #include <wiringPi.h> | ||||||
|  | @ -12,13 +11,13 @@ | ||||||
| 
 | 
 | ||||||
| // Globales
 | // Globales
 | ||||||
| pthread_t tIHM; | pthread_t tIHM; | ||||||
|  | pthread_t tStdinIHM; | ||||||
| 
 | 
 | ||||||
| // Fonctions
 | // Fonctions
 | ||||||
| void configureIHM() | void configureIHM() | ||||||
| { | { | ||||||
|     initLCD(); |     initLCD(); | ||||||
|     gotoLCD(LCD_LINE_1); |     printToLCD(LCD_LINE_1, "Demarrage"); | ||||||
|     printLCD("Demarrage..."); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void startIHM() | void startIHM() | ||||||
|  | @ -31,6 +30,7 @@ void startIHM() | ||||||
|     pullUpDnControl(IHM_PIN_TIRETTE, PUD_UP); |     pullUpDnControl(IHM_PIN_TIRETTE, PUD_UP); | ||||||
| 
 | 
 | ||||||
|     pthread_create(&tIHM, NULL, TaskIHM, NULL); |     pthread_create(&tIHM, NULL, TaskIHM, NULL); | ||||||
|  |     pthread_create(&tStdinIHM, NULL, TaskStdinIHM, NULL); | ||||||
|     pthread_join(tIHM, NULL); |     pthread_join(tIHM, NULL); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -66,11 +66,35 @@ bool debunkButtonIHM(int pin) | ||||||
|     return true; |     return true; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | enum boutons stdinbutton = none; | ||||||
|  | 
 | ||||||
|  | void* TaskStdinIHM(void* pdata) | ||||||
|  | { | ||||||
|  |     (void)pdata; | ||||||
|  | 
 | ||||||
|  |     for (;;) { | ||||||
|  |         char c = getchar(); | ||||||
|  |         if (c == '1') { | ||||||
|  |             stdinbutton = jaune; | ||||||
|  |         } else if (c == '2') { | ||||||
|  |             stdinbutton = rouge; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return NULL; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| enum boutons pressedIHM(int timeout) | enum boutons pressedIHM(int timeout) | ||||||
| { | { | ||||||
|     bool block = timeout < 0; |     bool block = timeout < 0; | ||||||
|     while (timeout > 0 || block) { |     while (timeout > 0 || block) { | ||||||
| 
 | 
 | ||||||
|  |         if (stdinbutton != none) { | ||||||
|  |             enum boutons bout = stdinbutton; | ||||||
|  |             stdinbutton = none; | ||||||
|  |             return bout; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         if (debunkButtonIHM(IHM_PIN_JAUNE)) { |         if (debunkButtonIHM(IHM_PIN_JAUNE)) { | ||||||
|             return jaune; |             return jaune; | ||||||
|         } |         } | ||||||
|  | @ -100,44 +124,33 @@ bool tirettePresente() | ||||||
|     return etat == LOW; |     return etat == LOW; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| char tempLine[16]; |  | ||||||
| bool isDebug = false; | bool isDebug = false; | ||||||
| bool isOrange = true; | bool isOrange = true; | ||||||
| bool annuler = false; | bool annuler = false; | ||||||
| clock_t lastCalibrage = 0; | clock_t lastCalibrage = 0; | ||||||
| clock_t parcoursStart = 0; |  | ||||||
| pthread_t tParcours; | pthread_t tParcours; | ||||||
| 
 | 
 | ||||||
| void printCouleur() | char* orangeStr = "Orange"; | ||||||
|  | char* vertStr = "Vert"; | ||||||
|  | 
 | ||||||
|  | char* getCouleur() | ||||||
| { | { | ||||||
|     if (isOrange) { |     return isOrange ? orangeStr : vertStr; | ||||||
|         printLCD("Orange"); |  | ||||||
|     } else { |  | ||||||
|         printLCD("Vert"); |  | ||||||
|     } |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void* TaskIHM(void* pdata) | void* TaskIHM(void* pdata) | ||||||
| { | { | ||||||
|     (void)pdata; |     (void)pdata; | ||||||
| 
 | 
 | ||||||
|     gotoLCD(LCD_LINE_1 + 1); |  | ||||||
|     printLCD("Niuh"); |  | ||||||
| 
 |  | ||||||
|     enum boutons bout; |     enum boutons bout; | ||||||
|     for (;;) { |     for (;;) { | ||||||
| 
 | 
 | ||||||
|         // Debug
 |         // Debug
 | ||||||
|         for (;;) { |         for (;;) { | ||||||
|             clearLCD(); |             clearLCD(); | ||||||
|             gotoLCD(LCD_LINE_1); |             printfToLCD(LCD_LINE_1, "Debug : %s", isDebug ? "On" : "Off"); | ||||||
|             printLCD("Debug : "); |  | ||||||
|             if (isDebug) { |             if (isDebug) { | ||||||
|                 printLCD("On"); |                 printToLCD(LCD_LINE_2, "192.168.0.0 TODO"); | ||||||
|                 gotoLCD(LCD_LINE_2); |  | ||||||
|                 printLCD("192.168.0.0 TODO"); |  | ||||||
|             } else { |  | ||||||
|                 printLCD("Off"); |  | ||||||
|             } |             } | ||||||
|             bout = pressedIHM(IHM_REFRESH_INTERVAL); |             bout = pressedIHM(IHM_REFRESH_INTERVAL); | ||||||
| 
 | 
 | ||||||
|  | @ -151,9 +164,7 @@ void* TaskIHM(void* pdata) | ||||||
|         // Couleur
 |         // Couleur
 | ||||||
|         for (;;) { |         for (;;) { | ||||||
|             clearLCD(); |             clearLCD(); | ||||||
|             gotoLCD(LCD_LINE_1); |             printfToLCD(LCD_LINE_1, "Couleur : %s", getCouleur()); | ||||||
|             printLCD("Couleur : "); |  | ||||||
|             printCouleur(); |  | ||||||
|             bout = pressedIHM(IHM_BLOCK); |             bout = pressedIHM(IHM_BLOCK); | ||||||
| 
 | 
 | ||||||
|             if (bout == rouge) { |             if (bout == rouge) { | ||||||
|  | @ -166,27 +177,20 @@ void* TaskIHM(void* pdata) | ||||||
|         // Calibrage
 |         // Calibrage
 | ||||||
|         for (;;) { |         for (;;) { | ||||||
|             clearLCD(); |             clearLCD(); | ||||||
|             gotoLCD(LCD_LINE_1); |  | ||||||
|             if (lastCalibrage != 0) { |             if (lastCalibrage != 0) { | ||||||
|                 printLCD("Calibre il y a"); |                 printToLCD(LCD_LINE_1, "Calibre il y a"); | ||||||
|                 gotoLCD(LCD_LINE_2); |                 printfToLCD(LCD_LINE_2, "%ld secondes", (clock() - lastCalibrage) / CLOCKS_PER_SEC); | ||||||
|                 sprintf(tempLine, "%ld secondes", (clock() - lastCalibrage) / CLOCKS_PER_SEC); |  | ||||||
|                 printLCD(tempLine); |  | ||||||
|             } else { |             } else { | ||||||
|                 printLCD("Calibrer"); |                 printToLCD(LCD_LINE_1, "Calibrer"); | ||||||
|                 gotoLCD(LCD_LINE_2); |                 printfToLCD(LCD_LINE_2, "(%s)", getCouleur()); | ||||||
|                 printLCD("("); |  | ||||||
|                 printCouleur(); |  | ||||||
|                 printLCD(")"); |  | ||||||
|             } |             } | ||||||
|             bout = pressedIHM(IHM_REFRESH_INTERVAL); |             bout = pressedIHM(IHM_REFRESH_INTERVAL); | ||||||
| 
 | 
 | ||||||
|             if (bout == rouge) { |             if (bout == rouge) { | ||||||
|                 clearLCD(); |                 clearLCD(); | ||||||
|                 gotoLCD(LCD_LINE_1); |                 printToLCD(LCD_LINE_1, "Calibrage..."); | ||||||
|                 printLCD("Calibrage..."); |                 delay(3000);             // TODO
 | ||||||
|                 delay(3000); // TODO
 |                 lastCalibrage = clock(); // TODO struct timespec
 | ||||||
|                 lastCalibrage = clock(); |  | ||||||
|             } else if (bout == jaune) { |             } else if (bout == jaune) { | ||||||
|                 break; |                 break; | ||||||
|             } |             } | ||||||
|  | @ -195,14 +199,12 @@ void* TaskIHM(void* pdata) | ||||||
|         // Diagnostics
 |         // Diagnostics
 | ||||||
|         for (;;) { |         for (;;) { | ||||||
|             clearLCD(); |             clearLCD(); | ||||||
|             gotoLCD(LCD_LINE_1); |             printToLCD(LCD_LINE_1, "Diagnostiquer"); | ||||||
|             printLCD("Diagnostiquer"); |  | ||||||
|             bout = pressedIHM(IHM_BLOCK); |             bout = pressedIHM(IHM_BLOCK); | ||||||
| 
 | 
 | ||||||
|             if (bout == rouge) { |             if (bout == rouge) { | ||||||
|                 clearLCD(); |                 clearLCD(); | ||||||
|                 gotoLCD(LCD_LINE_1); |                 printToLCD(LCD_LINE_1, "Diagnostics..."); | ||||||
|                 printLCD("Diagnostics..."); |  | ||||||
|                 delay(3000); // TODO
 |                 delay(3000); // TODO
 | ||||||
|             } else if (bout == jaune) { |             } else if (bout == jaune) { | ||||||
|                 break; |                 break; | ||||||
|  | @ -212,22 +214,16 @@ void* TaskIHM(void* pdata) | ||||||
|         // Parcours
 |         // Parcours
 | ||||||
|         for (;;) { |         for (;;) { | ||||||
|             clearLCD(); |             clearLCD(); | ||||||
|             gotoLCD(LCD_LINE_1); |             printToLCD(LCD_LINE_1, "Lancer parcours"); | ||||||
|             printLCD("Lancer parcours"); |             printfToLCD(LCD_LINE_2, "(%s)", getCouleur()); | ||||||
|             gotoLCD(LCD_LINE_2); |  | ||||||
|             printLCD("("); |  | ||||||
|             printCouleur(); |  | ||||||
|             printLCD(")"); |  | ||||||
|             bout = pressedIHM(IHM_BLOCK); |             bout = pressedIHM(IHM_BLOCK); | ||||||
|             if (bout == rouge) { |             if (bout == rouge) { | ||||||
|                 // No tirette
 |                 // No tirette
 | ||||||
|                 annuler = false; |                 annuler = false; | ||||||
|                 while (!tirettePresente()) { |                 while (!tirettePresente()) { | ||||||
|                     clearLCD(); |                     clearLCD(); | ||||||
|                     gotoLCD(LCD_LINE_1); |                     printToLCD(LCD_LINE_1, "Inserez tirette"); | ||||||
|                     printLCD("Inserez tirette"); |                     printToLCD(LCD_LINE_2, "(ROUGE: ignorer)"); | ||||||
|                     gotoLCD(LCD_LINE_2); |  | ||||||
|                     printLCD("(ROUGE: ignorer)"); |  | ||||||
|                     bout = pressedIHM(IHM_REFRESH_INTERVAL); |                     bout = pressedIHM(IHM_REFRESH_INTERVAL); | ||||||
|                     if (bout == rouge) { |                     if (bout == rouge) { | ||||||
|                         break; |                         break; | ||||||
|  | @ -277,14 +273,12 @@ void* TaskIHM(void* pdata) | ||||||
|         // RàZ
 |         // RàZ
 | ||||||
|         for (;;) { |         for (;;) { | ||||||
|             clearLCD(); |             clearLCD(); | ||||||
|             gotoLCD(LCD_LINE_1); |             printToLCD(LCD_LINE_1, "Remettre a zero"); | ||||||
|             printLCD("Remettre a zero"); |  | ||||||
|             bout = pressedIHM(IHM_BLOCK); |             bout = pressedIHM(IHM_BLOCK); | ||||||
| 
 | 
 | ||||||
|             if (bout == rouge) { |             if (bout == rouge) { | ||||||
|                 clearLCD(); |                 clearLCD(); | ||||||
|                 gotoLCD(LCD_LINE_1); |                 printToLCD(LCD_LINE_1, "Remise a zero..."); | ||||||
|                 printLCD("Remise a zero..."); |  | ||||||
|                 delay(3000); // TODO
 |                 delay(3000); // TODO
 | ||||||
|             } else if (bout == jaune) { |             } else if (bout == jaune) { | ||||||
|                 break; |                 break; | ||||||
|  | @ -296,6 +290,5 @@ void* TaskIHM(void* pdata) | ||||||
| void deconfigureIHM() | void deconfigureIHM() | ||||||
| { | { | ||||||
|     clearLCD(); |     clearLCD(); | ||||||
|     gotoLCD(LCD_LINE_1); |     printToLCD(LCD_LINE_1, "Bye bye!"); | ||||||
|     printLCD("Bye bye!"); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -1,6 +1,8 @@ | ||||||
| #ifndef __IHM_H_ | #ifndef __IHM_H_ | ||||||
| #define __IHM_H_ | #define __IHM_H_ | ||||||
| 
 | 
 | ||||||
|  | #include <stdbool.h> | ||||||
|  | 
 | ||||||
| #include "lcd.h" | #include "lcd.h" | ||||||
| 
 | 
 | ||||||
| #define IHM_PIN_ROUGE 0 | #define IHM_PIN_ROUGE 0 | ||||||
|  | @ -22,6 +24,8 @@ void deconfigureIHM(); | ||||||
| 
 | 
 | ||||||
| // Private
 | // Private
 | ||||||
| void* TaskIHM(void *pdata); | void* TaskIHM(void *pdata); | ||||||
|  | void* TaskStdinIHM(void *pdata); | ||||||
| enum boutons pressedIHM(int timeout); // timeout: ms or -1
 | enum boutons pressedIHM(int timeout); // timeout: ms or -1
 | ||||||
|  | bool debunkButtonIHM(int pin); | ||||||
| 
 | 
 | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | @ -1,4 +1,7 @@ | ||||||
| #include <pthread.h> | #include <pthread.h> | ||||||
|  | #include <stdarg.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <string.h> | ||||||
| #include <wiringPi.h> | #include <wiringPi.h> | ||||||
| #include <wiringPiI2C.h> | #include <wiringPiI2C.h> | ||||||
| 
 | 
 | ||||||
|  | @ -7,6 +10,9 @@ | ||||||
| int lcdFd; | int lcdFd; | ||||||
| pthread_mutex_t sLCD; | pthread_mutex_t sLCD; | ||||||
| 
 | 
 | ||||||
|  | char virtual[LCD_NB_TOTAL]; | ||||||
|  | uint8_t pos; | ||||||
|  | 
 | ||||||
| void initLCD() | void initLCD() | ||||||
| { | { | ||||||
|     lcdFd = openI2C(LCD_ADDR); |     lcdFd = openI2C(LCD_ADDR); | ||||||
|  | @ -17,23 +23,41 @@ void initLCD() | ||||||
|     sendLCD(0x06, LCD_MODE_CMD); // Cursor move direction
 |     sendLCD(0x06, LCD_MODE_CMD); // Cursor move direction
 | ||||||
|     sendLCD(0x0C, LCD_MODE_CMD); // Blink Off
 |     sendLCD(0x0C, LCD_MODE_CMD); // Blink Off
 | ||||||
|     sendLCD(0x28, LCD_MODE_CMD); // Data length, number of lines, font size
 |     sendLCD(0x28, LCD_MODE_CMD); // Data length, number of lines, font size
 | ||||||
|     sendLCD(0x01, LCD_MODE_CMD); // Clear display
 |     clearLCD(); | ||||||
|  | 
 | ||||||
|     delayMicroseconds(500); |     delayMicroseconds(500); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void clearLCD() | void clearLCD() | ||||||
| { | { | ||||||
|  |     lockLCD(); | ||||||
|     sendLCD(0x01, LCD_MODE_CMD); |     sendLCD(0x01, LCD_MODE_CMD); | ||||||
|     sendLCD(0x02, LCD_MODE_CMD); |     sendLCD(0x02, LCD_MODE_CMD); | ||||||
|  | 
 | ||||||
|  |     for (int i = 0; i < LCD_NB_TOTAL; i++) { | ||||||
|  |         virtual[i] = ' '; | ||||||
|  |     } | ||||||
|  |     unlockLCD(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void gotoLCD(uint8_t line) | void gotoLCD(uint8_t line) | ||||||
| { | { | ||||||
|  |     if (line >= LCD_LINE_2) { | ||||||
|  |         pos = 1 * LCD_NB_CHARS + line - LCD_LINE_2; | ||||||
|  |     } else if (line >= LCD_LINE_1) { | ||||||
|  |         pos = 0 * LCD_NB_CHARS + line - LCD_LINE_1; | ||||||
|  |     } else { | ||||||
|  |         pos = LCD_NB_TOTAL; | ||||||
|  |     } | ||||||
|     sendLCD(line, LCD_MODE_CMD); |     sendLCD(line, LCD_MODE_CMD); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void charLCD(char c) | void charLCD(char c) | ||||||
| { | { | ||||||
|  |     if (pos >= 0 && pos < LCD_NB_TOTAL) { | ||||||
|  |         virtual[pos] = c; | ||||||
|  |     } | ||||||
|  |     pos++; | ||||||
|     sendLCD(c, LCD_MODE_CHR); |     sendLCD(c, LCD_MODE_CHR); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | @ -45,6 +69,51 @@ void printLCD(char* s) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void displayLCD() | ||||||
|  | { | ||||||
|  |     printf("\n┌"); | ||||||
|  |     for (int i = 0; i < LCD_NB_CHARS; i++) { | ||||||
|  |         printf("─"); | ||||||
|  |     } | ||||||
|  |     printf("┐\n"); | ||||||
|  |     for (int y = 0; y < LCD_NB_LINES; y++) { | ||||||
|  |         printf("│"); | ||||||
|  |         for (int x = 0; x < LCD_NB_CHARS; x++) { | ||||||
|  |             putchar(virtual[y * LCD_NB_CHARS + x]); | ||||||
|  |         } | ||||||
|  |         printf("│\n"); | ||||||
|  |     } | ||||||
|  |     printf("└"); | ||||||
|  |     for (int i = 0; i < LCD_NB_CHARS; i++) { | ||||||
|  |         printf("─"); | ||||||
|  |     } | ||||||
|  |     printf("┘\n"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void printToLCD(uint8_t line, char* s) | ||||||
|  | { | ||||||
|  |     lockLCD(); | ||||||
|  |     gotoLCD(line); | ||||||
|  |     printLCD(s); | ||||||
|  |     displayLCD(); | ||||||
|  |     unlockLCD(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void printRightLCD(uint8_t line, char* s) | ||||||
|  | { | ||||||
|  |     printToLCD(line + LCD_NB_CHARS - strlen(s), s); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void printfToLCD(uint8_t line, char* s, ...) | ||||||
|  | { | ||||||
|  |     char buffer[LCD_NB_TOTAL]; | ||||||
|  |     va_list args; | ||||||
|  |     va_start(args, s); | ||||||
|  |     vsnprintf(buffer, LCD_NB_TOTAL, s, args); | ||||||
|  |     va_end(args); | ||||||
|  |     printToLCD(line, buffer); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void sendLCD(uint8_t bits, uint8_t mode) | void sendLCD(uint8_t bits, uint8_t mode) | ||||||
| { | { | ||||||
|     lockI2C(); |     lockI2C(); | ||||||
|  |  | ||||||
|  | @ -9,6 +9,11 @@ | ||||||
| #define LCD_MODE_CHR 1 | #define LCD_MODE_CHR 1 | ||||||
| #define LCD_MODE_CMD 0 | #define LCD_MODE_CMD 0 | ||||||
| 
 | 
 | ||||||
|  | #define LCD_NB_LINES 2 | ||||||
|  | #define LCD_NB_CHARS 16 | ||||||
|  | #define LCD_NB_TOTAL LCD_NB_LINES*LCD_NB_CHARS | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| #define LCD_LINE_1 0x80 // 1st line
 | #define LCD_LINE_1 0x80 // 1st line
 | ||||||
| #define LCD_LINE_2 0xC0 // 2nd line
 | #define LCD_LINE_2 0xC0 // 2nd line
 | ||||||
| 
 | 
 | ||||||
|  | @ -21,15 +26,18 @@ | ||||||
| // Public
 | // Public
 | ||||||
| void initLCD(); | void initLCD(); | ||||||
| void clearLCD(); | void clearLCD(); | ||||||
| void gotoLCD(uint8_t line); | void printToLCD(uint8_t line, char* s); | ||||||
| void charLCD(char c); | void printRightLCD(uint8_t line, char* s); | ||||||
| void printLCD(char* s); | void printfToLCD(uint8_t line, char* s, ...); | ||||||
| // Not necessary, but should be used when different threads use the display
 |  | ||||||
| void lockLCD(); |  | ||||||
| void unlockLCD(); |  | ||||||
| 
 | 
 | ||||||
| // Private
 | // Private
 | ||||||
| void sendLCD(uint8_t bits, uint8_t mode); | void sendLCD(uint8_t bits, uint8_t mode); | ||||||
| void toggleEnableLCD(uint8_t bits); | void toggleEnableLCD(uint8_t bits); | ||||||
|  | void gotoLCD(uint8_t line); | ||||||
|  | void charLCD(char c); | ||||||
|  | void printLCD(char* s); | ||||||
|  | void displayLCD(); | ||||||
|  | void lockLCD(); | ||||||
|  | void unlockLCD(); | ||||||
| 
 | 
 | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | @ -11,8 +11,8 @@ | ||||||
| #define PWM_MAX 1023 | #define PWM_MAX 1023 | ||||||
| #define PWM_MAX_V 3.3 | #define PWM_MAX_V 3.3 | ||||||
| 
 | 
 | ||||||
| // #define TESTINATOR
 | #define TESTINATOR | ||||||
| #define TLE5206 | // #define TLE5206
 | ||||||
| 
 | 
 | ||||||
| #ifdef TESTINATOR | #ifdef TESTINATOR | ||||||
| #define MOT_MIN_V 0.1 | #define MOT_MIN_V 0.1 | ||||||
|  | @ -26,16 +26,14 @@ | ||||||
| 
 | 
 | ||||||
| // Pins definition
 | // Pins definition
 | ||||||
| // Left
 | // Left
 | ||||||
| // Physical 32
 |  | ||||||
| #define ENA 26 | #define ENA 26 | ||||||
| #define IN1 2 | #define IN1 21 | ||||||
| #define IN2 3 | #define IN2 22 | ||||||
| 
 | 
 | ||||||
| // Physical 33
 |  | ||||||
| // Right
 | // Right
 | ||||||
| #define ENB 23 | #define ENB 23 | ||||||
| #define IN3 4 | #define IN3 24 | ||||||
| #define IN4 5 | #define IN4 25 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| void configureMovement(); | void configureMovement(); | ||||||
|  |  | ||||||
|  | @ -1,16 +1,15 @@ | ||||||
| #include <unistd.h> |  | ||||||
| #include <pthread.h> | #include <pthread.h> | ||||||
| #include <stdio.h> | #include <stdio.h> | ||||||
|  | #include <unistd.h> | ||||||
| 
 | 
 | ||||||
| #include "parcours.h" |  | ||||||
| #include "movement.h" |  | ||||||
| #include "position.h" |  | ||||||
| #include "points.h" |  | ||||||
| #include "lcd.h" | #include "lcd.h" | ||||||
|  | #include "movement.h" | ||||||
|  | #include "parcours.h" | ||||||
|  | #include "points.h" | ||||||
|  | #include "position.h" | ||||||
| 
 | 
 | ||||||
| pthread_t tParcours; | pthread_t tParcours; | ||||||
| bool isOrange; | bool isOrange; | ||||||
| char tempLine[16]; |  | ||||||
| struct timespec tempsStart; | struct timespec tempsStart; | ||||||
| struct timespec tempsNow; | struct timespec tempsNow; | ||||||
| struct timespec tempsEcoule; | struct timespec tempsEcoule; | ||||||
|  | @ -19,35 +18,24 @@ void prepareParcours(bool orange) | ||||||
| { | { | ||||||
|     isOrange = orange; |     isOrange = orange; | ||||||
|     clearLCD(); |     clearLCD(); | ||||||
|     gotoLCD(LCD_LINE_1); |     printfToLCD(LCD_LINE_1, "--:--/%2d:%02d", TEMPS_PARCOURS / 60, TEMPS_PARCOURS % 60); | ||||||
|     sprintf(tempLine, "--:--/%2d:%02d", TEMPS_PARCOURS / 60, TEMPS_PARCOURS % 60); |     printRightLCD(LCD_LINE_1, "ATT"); | ||||||
|     printLCD(tempLine); |  | ||||||
|     gotoLCD(LCD_LINE_1 + 16 - 3); |  | ||||||
|     printLCD("ATT"); |  | ||||||
| 
 | 
 | ||||||
|     resetPoints(); |     resetPoints(); | ||||||
|     showPoints(); |     showPoints(); | ||||||
|     gotoLCD(LCD_LINE_2 + 16 - 3); |     printRightLCD(LCD_LINE_2, isOrange ? "Org" : "Vrt"); | ||||||
|     printLCD(isOrange ? "Org" : "Vrt"); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void startParcours() | void startParcours() | ||||||
| { | { | ||||||
|     clock_gettime(CLOCK_REALTIME, &tempsStart); |     clock_gettime(CLOCK_REALTIME, &tempsStart); | ||||||
|     pthread_create(&tParcours, NULL, TaskParcours, NULL); // TODO Start on mutex unlock
 |     pthread_create(&tParcours, NULL, TaskParcours, NULL); // TODO Start on mutex unlock
 | ||||||
|     lockLCD(); |     printRightLCD(LCD_LINE_1, "   "); | ||||||
|     gotoLCD(LCD_LINE_1 + 16 - 3); |  | ||||||
|     printLCD("   "); |  | ||||||
|     unlockLCD(); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void updateTimeDisplay() | void updateTimeDisplay() | ||||||
| { | { | ||||||
|     lockLCD(); |     printfToLCD(LCD_LINE_1, "%2ld:%02ld", tempsEcoule.tv_sec / 60, tempsEcoule.tv_sec % 60); | ||||||
|     gotoLCD(LCD_LINE_1); |  | ||||||
|     sprintf(tempLine, "%2ld:%02ld", tempsEcoule.tv_sec / 60, tempsEcoule.tv_sec % 60); |  | ||||||
|     printLCD(tempLine); |  | ||||||
|     unlockLCD(); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int updateParcours() | int updateParcours() | ||||||
|  | @ -76,62 +64,43 @@ void stopParcours() | ||||||
|     stop(); |     stop(); | ||||||
| 
 | 
 | ||||||
|     updateTimeDisplay(); |     updateTimeDisplay(); | ||||||
|     gotoLCD(LCD_LINE_1 + 16 - 3); |     printRightLCD(LCD_LINE_1, "FIN"); | ||||||
|     printLCD("FIN"); |  | ||||||
|     showPoints(); |     showPoints(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | #define UP_TIME 1000 | ||||||
|  | #define HIGH_TIME 30000 | ||||||
|  | #define DOWN_TIME 1000 | ||||||
|  | #define LOW_TIME 2000 | ||||||
|  | #define MAX_VIT MOT_MAX_V | ||||||
|  | 
 | ||||||
| void* TaskParcours(void* pdata) | void* TaskParcours(void* pdata) | ||||||
| { | { | ||||||
|     (void)pdata; |     (void)pdata; | ||||||
| 
 | 
 | ||||||
|     for (;;) { |     for (;;) { | ||||||
|         delay(250); |  | ||||||
|         addPoints(1); |         addPoints(1); | ||||||
|  |         for (int i = 0; i < UP_TIME; i++) { | ||||||
|  |             float p = (float)i / (float)UP_TIME; | ||||||
|  |             changerMoteurs(p * MOT_MAX_V, p * MOT_MAX_V); | ||||||
|  |             delay(1); | ||||||
|  |         } | ||||||
|  |         addPoints(1); | ||||||
|  |         changerMoteurs(MOT_MAX_V, MOT_MAX_V); | ||||||
|  |         delay(HIGH_TIME); | ||||||
|  | 
 | ||||||
|  |         addPoints(1); | ||||||
|  |         for (int i = 0; i < DOWN_TIME; i++) { | ||||||
|  |             float p = (float)i / (float)DOWN_TIME; | ||||||
|  |             p = 1 - p; | ||||||
|  |             changerMoteurs(p * MOT_MAX_V, p * MOT_MAX_V); | ||||||
|  |             delay(1); | ||||||
|  |         } | ||||||
|  |         addPoints(1); | ||||||
|  |         changerMoteurs(0, 0); | ||||||
|  |         delay(LOW_TIME); | ||||||
|     } |     } | ||||||
| } |  | ||||||
| 
 |  | ||||||
| void* TaskParcours2(void* pdata) |  | ||||||
| { |  | ||||||
|     (void)pdata; |  | ||||||
| 
 |  | ||||||
|     /* struct position pos; */ |  | ||||||
|     /* for (;;) { */ |  | ||||||
|     /*     pos.x = (int) (rand()*200.0/RAND_MAX); */ |  | ||||||
|     /*     pos.y = (int) (rand()*100.0/RAND_MAX); */ |  | ||||||
|     /*     pos.o = (int) (rand()*360.0/RAND_MAX); */ |  | ||||||
|     /*     aller(&pos); */ |  | ||||||
|     /*     sleep(1); */ |  | ||||||
|     /*     brake(); */ |  | ||||||
|     /*     sleep(2); */ |  | ||||||
|     /* } */ |  | ||||||
| 
 |  | ||||||
|     struct timespec tim; // 10 ms
 |  | ||||||
|     tim.tv_sec = 0; |  | ||||||
|     tim.tv_nsec = 10000000L; |  | ||||||
| 
 |  | ||||||
| #define RAMP_TIME 100 |  | ||||||
| #define MAX_VIT MOT_MAX_V |  | ||||||
| 
 |  | ||||||
|     /* for (;;) { */ |  | ||||||
|     // ↗
 |  | ||||||
|     for (int i = 0; i < RAMP_TIME; i++) { |  | ||||||
|         float p = (float)i / (float)RAMP_TIME; |  | ||||||
|         changerMoteurs(p * MOT_MAX_V, p * MOT_MAX_V); |  | ||||||
|         nanosleep(&tim, NULL); |  | ||||||
|     } |  | ||||||
|     changerMoteurs(MOT_MAX_V, MOT_MAX_V); |  | ||||||
|     // ↑
 |  | ||||||
|     sleep(2); |  | ||||||
|     /*     // ↘ */ |  | ||||||
|     /*     for (int i = 0; i < RAMP_TIME; i++) { */ |  | ||||||
|     /*         float p = (float) i / (float) RAMP_TIME; */ |  | ||||||
|     /*         p = 1 - p; */ |  | ||||||
|     /*         changerMoteurs(p * MOT_MAX_V, p * MOT_MAX_V); */ |  | ||||||
|     /*         nanosleep(&tim, NULL); */ |  | ||||||
|     /*     } */ |  | ||||||
|     /*     sleep(5); */ |  | ||||||
|     /* } */ |  | ||||||
| 
 | 
 | ||||||
|     return NULL; |     return NULL; | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  |  | ||||||
|  | @ -3,7 +3,6 @@ | ||||||
| #include "lcd.h" | #include "lcd.h" | ||||||
| 
 | 
 | ||||||
| int points; | int points; | ||||||
| char tempLine[16]; |  | ||||||
| 
 | 
 | ||||||
| void resetPoints() | void resetPoints() | ||||||
| { | { | ||||||
|  | @ -17,11 +16,7 @@ int getPoints() | ||||||
| 
 | 
 | ||||||
| void showPoints() | void showPoints() | ||||||
| { | { | ||||||
|     sprintf(tempLine, "%d points", getPoints()); |     printfToLCD(LCD_LINE_2, "%d points", getPoints()); | ||||||
|     lockLCD(); |  | ||||||
|     gotoLCD(LCD_LINE_2); |  | ||||||
|     printLCD(tempLine); |  | ||||||
|     unlockLCD(); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void addPoints(int pts) | void addPoints(int pts) | ||||||
|  |  | ||||||
							
								
								
									
										31
									
								
								chef/src/testForward.c
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								chef/src/testForward.c
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,31 @@ | ||||||
|  | /* Teste si une broche est connecté à une autre */ | ||||||
|  | 
 | ||||||
|  | #include <stdint.h> | ||||||
|  | #include <stdio.h> | ||||||
|  | #include <stdlib.h> | ||||||
|  | #include <unistd.h> | ||||||
|  | #include <wiringPiI2C.h> | ||||||
|  | 
 | ||||||
|  | #include "lcd.h" | ||||||
|  | #include "movement.h" | ||||||
|  | 
 | ||||||
|  | int main(int argc, char* argv[]) | ||||||
|  | { | ||||||
|  | 
 | ||||||
|  |     (void)argc; | ||||||
|  |     (void)argv; | ||||||
|  | 
 | ||||||
|  |     initI2C(); | ||||||
|  | 
 | ||||||
|  |     initLCD(); | ||||||
|  |     clearLCD(); | ||||||
|  |     printToLCD(LCD_LINE_1, "Forward"); | ||||||
|  | 
 | ||||||
|  |     configureMovement(); | ||||||
|  | 
 | ||||||
|  |     for (;;) { | ||||||
|  |         changerMoteurs(1.5, 2); | ||||||
|  |         sleep(1); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | @ -6,7 +6,6 @@ | ||||||
| #include <unistd.h> | #include <unistd.h> | ||||||
| #include <wiringPiI2C.h> | #include <wiringPiI2C.h> | ||||||
| 
 | 
 | ||||||
| #include "srf08.h" |  | ||||||
| #include "i2c.h" | #include "i2c.h" | ||||||
| #include "lcd.h" | #include "lcd.h" | ||||||
| 
 | 
 | ||||||
|  | @ -17,34 +16,9 @@ int main(int argc, char* argv[]) | ||||||
|     (void)argv; |     (void)argv; | ||||||
| 
 | 
 | ||||||
|     initI2C(); |     initI2C(); | ||||||
| 
 |  | ||||||
|     // Adresse du SRF05
 |  | ||||||
|     uint8_t address = atoi(argv[1]); |  | ||||||
| 
 |  | ||||||
|     printf("Connecting to %d (0x%x)\n", address, address); |  | ||||||
| 
 |  | ||||||
|     initLCD(); |     initLCD(); | ||||||
|     clearLCD(); |     clearLCD(); | ||||||
|     gotoLCD(LCD_LINE_1); |     gotoLCD(LCD_LINE_1); | ||||||
|     printLCD("Distance"); |     printLCD("Bojour"); | ||||||
| 
 |  | ||||||
|     int front = openSRF08(address); |  | ||||||
| 
 |  | ||||||
|     char line[16]; |  | ||||||
|     float l = 0; |  | ||||||
|     for (;;) { |  | ||||||
|         startSRF08(front); |  | ||||||
| 
 |  | ||||||
|         gotoLCD(LCD_LINE_2); |  | ||||||
|         printf("l = %3f\n", l); |  | ||||||
|         sprintf(line, "%f mm", l); |  | ||||||
|         printLCD(line); |  | ||||||
|         printLCD("        "); |  | ||||||
| 
 |  | ||||||
|         waitSRF08(front); |  | ||||||
|         l = getSRF08(front); |  | ||||||
| 
 |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -91,7 +91,8 @@ upgrade-filesystem: sshconf configure | ||||||
| # Met jour les overlays (une partie des fichiers)
 | # Met jour les overlays (une partie des fichiers)
 | ||||||
| upgrade-overlays: sshconf | upgrade-overlays: sshconf | ||||||
| 	ssh -F sshconf principal true | 	ssh -F sshconf principal true | ||||||
| 	rsync --rsh 'ssh -F sshconf' --archive --chown root:root robotech/chef/rootfs_overlay/ principal:/ | 	chmod 700 board/robotech/cdfprincipal/rootfs_overlay/root | ||||||
|  | 	rsync --rsh 'ssh -F sshconf' --archive --chown root:root board/robotech/cdfprincipal/rootfs_overlay/ principal:/ | ||||||
| 
 | 
 | ||||||
| # ARDUINO
 | # ARDUINO
 | ||||||
| upgrade-arduino: | upgrade-arduino: | ||||||
|  | @ -116,6 +117,11 @@ upgrade-chef: chef | ||||||
| 	rsync --rsh 'ssh -F sshconf' --archive --chown root:root buildroot/output/target/opt/chef principal:/opt/ | 	rsync --rsh 'ssh -F sshconf' --archive --chown root:root buildroot/output/target/opt/chef principal:/opt/ | ||||||
| 
 | 
 | ||||||
| run: | run: | ||||||
|  | 	ssh -F sshconf principal true | ||||||
|  | 	ssh -F sshconf principal /etc/init.d/S50chef stop | ||||||
|  | 	ssh -F sshconf principal 'cd /opt/chef; bin/premier; /opt/chef/lcdOff.sh' | ||||||
|  | 
 | ||||||
|  | restart: | ||||||
| 	ssh -F sshconf principal true | 	ssh -F sshconf principal true | ||||||
| 	ssh -F sshconf principal /etc/init.d/S50chef restart | 	ssh -F sshconf principal /etc/init.d/S50chef restart | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -8,7 +8,7 @@ start() { | ||||||
| 	modprobe pl2303 # USB↔Serial cable | 	modprobe pl2303 # USB↔Serial cable | ||||||
| 	modprobe i2c-bcm2708 # I2C | 	modprobe i2c-bcm2708 # I2C | ||||||
| 	modprobe i2c-dev # I2C | 	modprobe i2c-dev # I2C | ||||||
| 	/opt/chef/i2cOff.sh | 	/opt/chef/lcdOff.sh | ||||||
| 	echo "OK" | 	echo "OK" | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -3,19 +3,22 @@ | ||||||
| # Start main program | # Start main program | ||||||
| # | # | ||||||
| 
 | 
 | ||||||
| EXEC=$(which sh) | SHELL=$(which sh) | ||||||
|  | EXECDIR=/opt/chef | ||||||
|  | EXECNAME=premier | ||||||
|  | EXECPATH=bin/$EXECNAME | ||||||
| PIDFILE=/var/run/chef.pid | PIDFILE=/var/run/chef.pid | ||||||
| LOGFILE=/var/run/chef.log |  | ||||||
| 
 | 
 | ||||||
| start() { | start() { | ||||||
| 	printf "Starting chef: " | 	printf "Starting chef: " | ||||||
| 	start-stop-daemon -p "$PIDFILE" -x "$EXEC" -b -m -S -- "/opt/chef/run.sh" -l "$LOGFILE" | 	cd "$EXECDIR"; /sbin/start-stop-daemon -p "$PIDFILE" -x "$EXECPATH" -b -m -S | ||||||
| echo "OK" | 	echo "OK" | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| stop() { | stop() { | ||||||
| 	printf "Stopping chef: " | 	printf "Stopping chef: " | ||||||
| 	start-stop-daemon -p "$PIDFILE" -x "$EXEC" -K | 	/sbin/start-stop-daemon -x "$EXECPATH" -K | ||||||
|  | 	/opt/chef/lcdOff.sh | ||||||
| 	echo "OK" | 	echo "OK" | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -5,10 +5,7 @@ | ||||||
| 
 | 
 | ||||||
| start() { | start() { | ||||||
| 	printf "Starting extra services: " | 	printf "Starting extra services: " | ||||||
| 	# if ! /opt/cdf/bin/testpin 0 1 | 	/etc/extra.d/rcS | ||||||
| 	# then |  | ||||||
| 		/etc/extra.d/rcS |  | ||||||
| 	# fi |  | ||||||
| 	echo "OK" | 	echo "OK" | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue