radioEmission.cpp 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197
  1. #include <wiringPi.h>
  2. #include <stdio.h>
  3. #include <sys/time.h>
  4. #include <time.h>
  5. #include <stdlib.h>
  6. #include <sched.h>
  7. #include <string.h>
  8. /*
  9. Par Idleman (idleman@idleman.fr - http://blog.idleman.fr)
  10. Licence : CC by sa
  11. Toutes question sur le blog ou par mail, possibilité de m'envoyer des bières via le blog
  12. */
  13. int pin;
  14. bool bit2[26]={}; // 26 bit Identifiant emetteur
  15. bool bit2Interruptor[4]={};
  16. int interruptor;
  17. int sender;
  18. char onoff;
  19. void scheduler_realtime()
  20. {
  21. struct sched_param p;
  22. p.__sched_priority = sched_get_priority_max(SCHED_RR);
  23. if (sched_setscheduler( 0, SCHED_RR, &p ) == -1 ) {
  24. perror("Failed to switch to realtime scheduler.");
  25. }
  26. }
  27. void scheduler_standard()
  28. {
  29. struct sched_param p;
  30. p.__sched_priority = 0;
  31. if( sched_setscheduler( 0, SCHED_OTHER, &p ) == -1 ) {
  32. perror("Failed to switch to normal scheduler.");
  33. }
  34. }
  35. //Envois d'une pulsation (passage de l'etat haut a l'etat bas)
  36. //1 = 310µs haut puis 1340µs bas
  37. //0 = 310µs haut puis 310µs bas
  38. void sendBit(bool b) {
  39. if (b) {
  40. digitalWrite(pin, HIGH);
  41. delayMicroseconds(310); //275 orinally, but tweaked.
  42. digitalWrite(pin, LOW);
  43. delayMicroseconds(1340); //1225 orinally, but tweaked.
  44. } else {
  45. digitalWrite(pin, HIGH);
  46. delayMicroseconds(310); //275 orinally, but tweaked.
  47. digitalWrite(pin, LOW);
  48. delayMicroseconds(310); //275 orinally, but tweaked.
  49. }
  50. }
  51. //Calcul le nombre 2^chiffre indiqué, fonction utilisé par itob pour la conversion decimal/binaire
  52. unsigned long power2(int power)
  53. {
  54. unsigned long integer=1;
  55. for (int i=0; i<power; i++){
  56. integer*=2;
  57. }
  58. return integer;
  59. }
  60. //Convertis un nombre en binaire, nécessite le nombre, et le nombre de bits souhaité en sortie (ici 26)
  61. // Stocke le résultat dans le tableau global "bit2"
  62. void itob(unsigned long integer, int length)
  63. {
  64. for (int i=0; i<length; i++){
  65. if ((integer / power2(length-1-i))==1) {
  66. integer-=power2(length-1-i);
  67. bit2[i]=1;
  68. } else {
  69. bit2[i]=0;
  70. }
  71. }
  72. }
  73. void itobInterruptor(unsigned long integer, int length)
  74. {
  75. for (int i=0; i<length; i++) {
  76. if ((integer / power2(length-1-i))==1) {
  77. integer-=power2(length-1-i);
  78. bit2Interruptor[i]=1;
  79. } else {
  80. bit2Interruptor[i]=0;
  81. }
  82. }
  83. }
  84. //Envoie d'une paire de pulsation radio qui definissent 1 bit réel : 0 =01 et 1 =10
  85. //c'est le codage de manchester qui necessite ce petit bouzin, ceci permet entre autres de dissocier les données des parasites
  86. void sendPair(bool b) {
  87. if(b) {
  88. sendBit(true);
  89. sendBit(false);
  90. } else {
  91. sendBit(false);
  92. sendBit(true);
  93. }
  94. }
  95. //Fonction d'envois du signal
  96. //recoit en parametre un booleen définissant l'arret ou la marche du matos (true = on, false = off)
  97. void transmit(int blnOn)
  98. {
  99. int i;
  100. // Sequence de verrou anoncant le départ du signal au recepeteur
  101. digitalWrite(pin, HIGH);
  102. delayMicroseconds(275); // un bit de bruit avant de commencer pour remettre les delais du recepteur a 0
  103. digitalWrite(pin, LOW);
  104. delayMicroseconds(9900); // premier verrou de 9900µs
  105. digitalWrite(pin, HIGH); // high again
  106. delayMicroseconds(275); // attente de 275µs entre les deux verrous
  107. digitalWrite(pin, LOW); // second verrou de 2675µs
  108. delayMicroseconds(2675);
  109. digitalWrite(pin, HIGH); // On reviens en état haut pour bien couper les verrous des données
  110. // Envoie du code emetteur (272946 = 1000010101000110010 en binaire)
  111. for(i=0; i<26;i++) {
  112. sendPair(bit2[i]);
  113. }
  114. // Envoie du bit définissant si c'est une commande de groupe ou non (26em bit)
  115. sendPair(false);
  116. // Envoie du bit définissant si c'est allumé ou eteint 27em bit)
  117. sendPair(blnOn);
  118. // Envoie des 4 derniers bits, qui représentent le code interrupteur, ici 0 (encode sur 4 bit donc 0000)
  119. // nb: sur les télécommandes officielle chacon, les interrupteurs sont logiquement nommés de 0 à x
  120. // interrupteur 1 = 0 (donc 0000) , interrupteur 2 = 1 (1000) , interrupteur 3 = 2 (0100) etc...
  121. for(i=0; i<4;i++) {
  122. if(bit2Interruptor[i]==0){
  123. sendPair(false);
  124. } else {
  125. sendPair(true);
  126. }
  127. }
  128. digitalWrite(pin, HIGH); // coupure données, verrou
  129. delayMicroseconds(275); // attendre 275µs
  130. digitalWrite(pin, LOW); // verrou 2 de 2675µs pour signaler la fermeture du signal
  131. delayMicroseconds(2675);
  132. }
  133. int main (int argc, char** argv)
  134. {
  135. pin = atoi(argv[1]);
  136. sender = atoi(argv[2]);
  137. interruptor = atoi(argv[3]);
  138. onoff = (strcmp(argv[4], "on")==0);
  139. //Si on ne trouve pas la librairie wiringPI, on arrête l'execution
  140. if (wiringPiSetup() == -1) {
  141. printf("Librairie Wiring PI introuvable, veuillez lier cette librairie...\n");
  142. return -1;
  143. }
  144. pinMode(pin, OUTPUT);
  145. printf("Pin GPIO configure en sortie\n");
  146. itob(sender,26); // convertion du code de l'emetteur (ici 8217034) en code binaire
  147. itobInterruptor(interruptor,4);
  148. scheduler_realtime();
  149. if(onoff){
  150. printf("envoi du signal ON\n");
  151. for(int i=0;i<5;i++){
  152. transmit(true); // envoyer ON
  153. }
  154. } else {
  155. printf("envoi du signal OFF\n");
  156. for(int i=0;i<5;i++){
  157. transmit(false); // envoyer OFF
  158. }
  159. }
  160. scheduler_standard();
  161. }