/* irmc - internet relay morsecode client */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include //#define DEBUG 1 #define MAXDATASIZE 1024 // max number of bytes we can get at once #include "cwprotocol.h" #include "beep.h" #include "util.h" //#define RASPI // http://raspberrypiguide.de/howtos/raspberry-pi-gpio-how-to/ #ifdef RASPI #include #endif int serial_status = 0, fd_serial, numbytes; double tx_timeout = 0; long tx_timer = 0; #define TX_WAIT 5000 #define TX_TIMEOUT 240.0 #define KEEPALIVE_CYCLE 100 /* TX Methods */ #define TX_NONE 0 #define TX_SERIAL 1 #define TX_KEYBOARD 2 #define TX_RASPI 3 long key_press_t1; long key_release_t1; int last_message = 0; char last_sender[16]; /* settings */ int translate = 1; int audio_status = 1; // disconnect from the server void inthandler(int sig) { signal(sig, SIG_IGN); send(fd_socket, &disconnect_packet, SIZE_COMMAND_PACKET, 0); close(fd_socket); close(fd_serial); exit(1); } void txloop (void) { key_press_t1 = fastclock(); tx_timeout = 0; for(;;){ tx_data_packet.n++; tx_data_packet.code[tx_data_packet.n - 1] = (int) ((key_press_t1 - key_release_t1) * -1); //printf("space: %i\n", tx_data_packet.code[tx_data_packet.n -1]); while(serial_status & TIOCM_DSR) ioctl(fd_serial, TIOCMGET, &serial_status); #ifdef RASPI while(digitalRead(5)==1) { } #endif key_release_t1 = fastclock(); tx_data_packet.n++; tx_data_packet.code[tx_data_packet.n - 1] = (int) ((key_release_t1 - key_press_t1) * 1); //printf("mark: %i\n", tx_data_packet.code[tx_data_packet.n -1]); while(1){ ioctl(fd_serial, TIOCMGET, &serial_status); if(serial_status & TIOCM_DSR) break; #ifdef RASPI if(digitalRead(5)==1) break; #endif tx_timeout = fastclock() - key_release_t1; if(tx_timeout > TX_TIMEOUT) return; } key_press_t1 = fastclock(); if(tx_data_packet.n == SIZE_CODE) { printf("irmc: warning packet is full.\n"); return; } } } void message(int msg) { switch(msg){ case 1: if(last_message == msg) return; if(last_message == 2) printf("\n"); last_message = msg; printf("irmc: transmitting...\n"); break; case 2: if(last_message == msg && strncmp(last_sender, rx_data_packet.id, 3) == 0) return; else { if(last_message == 2) printf("\n"); last_message = msg; strncpy(last_sender, rx_data_packet.id, 3); printf("irmc: receiving...(%s)\n", rx_data_packet.id); } break; case 3: printf("irmc: circuit was latched by %s.\n", rx_data_packet.id); break; case 4: printf("irmc: circuit was unlatched by %s.\n", rx_data_packet.id); break; default: break; } fflush(0); } /* Main Loop */ int main(int argc, char *argv[]) { char buf[MAXDATASIZE]; struct addrinfo hints, *servinfo, *p; int rv, i; char s[INET6_ADDRSTRLEN]; int keepalive_t = 0; char hostname[64]; char port[16]; int channel; char id[SIZE_ID]; char serialport[64]; int tx_method = TX_NONE; // Set default values snprintf(hostname, 64, "mtc-kob.dyndns.org"); snprintf(port, 16, "7890"); channel = 103; snprintf(id, SIZE_ID, "irmc-default"); snprintf(serialport, 64, ""); // Read commandline opterr = 0; int c; while ((c = getopt (argc, argv, "h:p:c:i:s:")) != -1) { switch (c) { case 'h': snprintf(hostname, 64, "%s", optarg); break; case 'p': snprintf(port, 16, "%s", optarg); break; case 'c': channel = atoi (optarg); break; case 'i': snprintf(id, SIZE_ID, "%s", optarg); break; case 's': snprintf(serialport, 64, "%s", optarg); break; case '?': fprintf(stderr, "irmc - Internet Relay Morse Code\n\n"); fprintf(stderr, "usage: irmc [arguments]\n\n"); fprintf(stderr, "Arguments:\n\n"); fprintf(stderr, " -h [hostname] Hostname of morsekob server. Default: %s\n", hostname); fprintf(stderr, " -p [port] Port of morsekob server. Default: %s\n", port); fprintf(stderr, " -c [channel] Channel. Default: %d\n", channel); fprintf(stderr, " -i [id] My ID. Default: %s\n", id); fprintf(stderr, " -s [serialport] Serial port device name. Example: /dev/tty.usbserial Default: \"%s\"\n", serialport); return 1; default: abort (); } } // Preparing connection fprintf(stderr, "irmc - Internet Relay Morse Code\n\n"); fprintf(stderr, "Connecting to %s:%s on channel %d with ID %s.\n", hostname, port, channel, id); prepare_id (&id_packet, id); prepare_tx (&tx_data_packet, id); connect_packet.channel = channel; signal(SIGINT, inthandler); memset(&hints, 0, sizeof hints); hints.ai_family = AF_UNSPEC; /* ipv4 or ipv6 */ hints.ai_socktype = SOCK_DGRAM; if ((rv = getaddrinfo(hostname, port, &hints, &servinfo)) != 0) { fprintf(stderr, "getaddrinfo: %s\n", gai_strerror(rv)); return 1; } /* Find the first free socket */ for(p = servinfo; p != NULL; p = p->ai_next) { if ((fd_socket = socket(p->ai_family, p->ai_socktype, p->ai_protocol)) == -1) { perror("irmc: socket"); continue; } if (connect(fd_socket, p->ai_addr, p->ai_addrlen) == -1) { close(fd_socket); perror("irmc: connect"); continue; } break; } fcntl(fd_socket, F_SETFL, O_NONBLOCK); if (p == NULL) { fprintf(stderr, "Failed to connect.\n"); return 2; } inet_ntop(p->ai_family, get_in_addr((struct sockaddr *)p->ai_addr), s, sizeof s); fprintf(stderr, "Connected to %s.\n", s); beep_init(); if ((strcmp (serialport, "")) != 0) tx_method = TX_SERIAL; if (tx_method == TX_SERIAL) { fd_serial = open(serialport, O_RDWR | O_NOCTTY | O_NDELAY); if(fd_serial == -1) { fprintf(stderr,"Unable to open serial port %s.\n", serialport); } } #ifdef RASPI // Starte die WiringPi-Api (wichtig) tx_method = TX_RASPI; if (wiringPiSetup() == -1) return 1; // Schalte GPIO 24 (=WiringPi Pin 5) auf Eingang pinMode(5, INPUT); //if (digitalRead(5)==1) #endif freeaddrinfo(servinfo); /* all done with this structure */ key_release_t1 = fastclock(); identifyclient(); /* Main Loop */ for(;;) { if(tx_timer == 0) if((numbytes = recv(fd_socket, buf, MAXDATASIZE-1, 0)) == -1) usleep(250); if(numbytes == SIZE_DATA_PACKET && tx_timer == 0){ memcpy(&rx_data_packet, buf, SIZE_DATA_PACKET); #if DEBUG printf("length: %i\n", rx_data_packet.length); printf("id: %s\n", rx_data_packet.id); printf("sequence no.: %i\n", rx_data_packet.sequence); printf("version: %s\n", rx_data_packet.status); printf("n: %i\n", rx_data_packet.n); printf("code:\n"); for(i = 0; i < SIZE_CODE; i++)printf("%i ", rx_data_packet.code[i]); printf("\n"); #endif if(rx_data_packet.n > 0 && rx_sequence != rx_data_packet.sequence){ message(2); if(translate == 1){ printf("%s", rx_data_packet.status); fflush(0); } rx_sequence = rx_data_packet.sequence; for(i = 0; i < rx_data_packet.n; i++){ switch(rx_data_packet.code[i]){ case 1: message(3); break; case 2: message(4); break; default: if(audio_status == 1) { int length = rx_data_packet.code[i]; if(length == 0 || abs(length) > 2000) { } else { if(length < 0) { beep(0.0, abs(length)/1000.); } else { beep(1000.0, length/1000.); } } } break; } } } } if(tx_timer > 0) tx_timer--; if(tx_data_packet.n > 1 ){ tx_sequence++; tx_data_packet.sequence = tx_sequence; for(i = 0; i < 5; i++) send(fd_socket, &tx_data_packet, SIZE_DATA_PACKET, 0); #if DEBUG printf("irmc: sent data packet.\n"); #endif tx_data_packet.n = 0; } if (tx_method == TX_SERIAL) { ioctl(fd_serial,TIOCMGET, &serial_status); if(serial_status & TIOCM_DSR){ txloop(); tx_timer = TX_WAIT; message(1); } } #ifdef RASPI if (tx_method == TX_RASPI) { if(digitalRead(5)==1){ txloop(); tx_timer = TX_WAIT; message(1); } } #endif if(keepalive_t < 0 && tx_timer == 0){ #if DEBUG printf("keep alive sent.\n"); #endif identifyclient(); keepalive_t = KEEPALIVE_CYCLE; } if(tx_timer == 0) { keepalive_t--; usleep(50); } if(kbhit() && tx_timer == 0){ getchar(); /* flush the buffer */ } } /* End of mainloop */ send(fd_socket, &disconnect_packet, SIZE_COMMAND_PACKET, 0); close(fd_socket); close(fd_serial); buzzer_stop(); exit(0); }