123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274 |
- /*====================================================================*
- *
- * Copyright (c) 2013 Qualcomm Atheros, Inc.
- *
- * All rights reserved.
- *
- *====================================================================*/
- /*====================================================================*
- *
- * hpavd.c - Qualcomm Atheros Homeplug AV Packet Daemon;
- *
- * prints incoming Qualcomm Atheros HomePlug AV vendor-specific
- * management messages on stdout in hex dump format for analysis;
- *
- * the program can be run in foreground or background, as a daemon;
- *
- * this program receives raw ethernet frames and so requires root
- * privileges; if you install it using "chmod 555" and "chown
- * root:root" then you must login as root to run it; otherwise, you
- * can install it using "chmod 4555" and "chown root:root" so that
- * anyone can run it; the program will refuse to run until you get
- * thing right;
- *
- * Contributor(s):
- * Charles Maier <cmaier@qca.qualcomm.com>
- *
- *--------------------------------------------------------------------*/
- /*====================================================================*
- * system header files;
- *--------------------------------------------------------------------*/
- #include <fcntl.h>
- #include <stdio.h>
- #include <ctype.h>
- #include <unistd.h>
- #include <stdint.h>
- #include <stdlib.h>
- #include <string.h>
- #include <sys/signal.h>
- #include <sys/ioctl.h>
- #include <errno.h>
- #ifdef __linux__
- #include <net/if.h>
- #include <net/if_arp.h>
- #include <netpacket/packet.h>
- #include <signal.h>
- #endif
- /*====================================================================*
- * custom header files;
- *--------------------------------------------------------------------*/
- #include "../tools/getoptv.h"
- #include "../tools/putoptv.h"
- #include "../tools/memory.h"
- #include "../tools/number.h"
- #include "../tools/types.h"
- #include "../tools/flags.h"
- #include "../tools/error.h"
- #include "../plc/plc.h"
- #include "../mme/mme.h"
- /*====================================================================*
- * custom source files;
- *--------------------------------------------------------------------*/
- #ifndef MAKEFILE
- #include "../tools/getoptv.c"
- #include "../tools/putoptv.c"
- #include "../tools/version.c"
- #include "../tools/hexdump.c"
- #include "../tools/hexstring.c"
- #include "../tools/hexdecode.c"
- #include "../tools/error.c"
- #include "../tools/emalloc.c"
- #endif
- #ifndef MAKEFILE
- #include "../mme/MMEPeek.c"
- #include "../mme/MMEName.c"
- #include "../mme/MMEMode.c"
- #endif
- /*====================================================================*
- * program constants;
- *--------------------------------------------------------------------*/
- #define HPAVD_DAEMON (1 << 0)
- #define HPAVD_VERBOSE (1 << 1)
- #define HPAVD_SILENCE (1 << 2)
- #define HPAVD_ERROR (1 << 3)
- #define PLCDEVICE "PLC"
- #define ETHDEVICE "eth1"
- #define ETHNUMBER 2
- /*====================================================================*
- * program variables;
- *--------------------------------------------------------------------*/
- static bool done = false;
- /*====================================================================*
- *
- * void terminate (signo_t signal);
- *
- * terminate the program; we want to ensure an orgaized program
- * exit so that the original Ethernet adapter state is restored;
- *
- *
- *--------------------------------------------------------------------*/
- static void terminate (signo_t signal)
- {
- done = true;
- return;
- }
- /*====================================================================*
- *
- * int main (int argc, char * argv[]);
- *
- *
- *--------------------------------------------------------------------*/
- int main (int argc, char const * argv [])
- {
- struct sigaction sa;
- struct ifreq ifreq;
- struct sockaddr_ll sockaddr =
- {
- PF_PACKET,
- htons (HOMEPLUG_MTYPE),
- 0x0000,
- ARPHRD_ETHER,
- PACKET_OTHERHOST,
- ETHER_ADDR_LEN,
- {
- 0x00,
- 0x00,
- 0x00,
- 0x00,
- 0x00,
- 0x00,
- 0x00,
- 0x00
- }
- };
- static char const * optv [] =
- {
- "di:qv",
- PUTOPTV_S_DIVINE,
- "Qualcomm Atheros HomePlug AV Packet Daemon",
- "d\trun in background as daemon",
- "i s\thost interface is (s) [" ETHDEVICE "]",
- "q\tsuppress normal output",
- "v\tverbose messages on stdout",
- (char const *) (0)
- };
- uint8_t packet [ETHER_MAX_LEN];
- flag_t state = (flag_t) (0);
- flag_t flags = (flag_t) (0);
- sock_t fd = - 1;
- signed c;
- memset (& ifreq, 0, sizeof (ifreq));
- memcpy (ifreq.ifr_name, ETHDEVICE, sizeof (ETHDEVICE));
- if (getenv (PLCDEVICE))
- {
- memcpy (ifreq.ifr_name, getenv (PLCDEVICE), sizeof (ifreq.ifr_name));
- }
- optind = 1;
- while (~ (c = getoptv (argc, argv, optv)))
- {
- switch ((char) (c))
- {
- case 'd':
- _setbits (flags, HPAVD_DAEMON);
- break;
- case 'i':
- memcpy (ifreq.ifr_name, optarg, sizeof (ifreq.ifr_name));
- break;
- case 'q':
- _setbits (flags, HPAVD_SILENCE);
- break;
- case 'v':
- _setbits (flags, HPAVD_VERBOSE);
- break;
- default:
- break;
- }
- }
- argc -= optind;
- argv += optind;
- if (geteuid ())
- {
- error (1, EPERM, ERROR_NOTROOT);
- }
- if (_anyset (flags, HPAVD_DAEMON))
- {
- pid_t pid = fork ();
- if (pid < 0)
- {
- error (1, errno, "razzlefrats!");
- }
- if (pid > 0)
- {
- exit (0);
- }
- }
- memset (& sa, 0, sizeof (struct sigaction));
- sa.sa_handler = terminate;
- sigaction (SIGTERM, & sa, (struct sigaction *) (0));
- sigaction (SIGQUIT, & sa, (struct sigaction *) (0));
- sigaction (SIGTSTP, & sa, (struct sigaction *) (0));
- sigaction (SIGINT, & sa, (struct sigaction *) (0));
- sigaction (SIGHUP, & sa, (struct sigaction *) (0));
- if ((fd = socket (sockaddr.sll_family, SOCK_RAW, sockaddr.sll_protocol)) == - 1)
- {
- error (1, errno, "Can't create socket for %s", ifreq.ifr_name);
- }
- if (ioctl (fd, SIOCGIFFLAGS, & ifreq) < 0)
- {
- error (1, errno, "Can't read %s device state", ifreq.ifr_name);
- }
- state = ifreq.ifr_flags;
- _setbits (ifreq.ifr_flags, (IFF_UP | IFF_BROADCAST));
- _clrbits (ifreq.ifr_flags, (IFF_MULTICAST | IFF_ALLMULTI | IFF_PROMISC));
- if (ioctl (fd, SIOCSIFFLAGS, & ifreq) < 0)
- {
- error (1, errno, "Can't change %s device state", ifreq.ifr_name);
- }
- if (ioctl (fd, SIOCGIFINDEX, & ifreq) == - 1)
- {
- error (1, errno, "Can't get %s interface index", ifreq.ifr_name);
- }
- sockaddr.sll_ifindex = ifreq.ifr_ifindex;
- if (ioctl (fd, SIOCGIFHWADDR, & ifreq) == - 1)
- {
- error (1, errno, "Can't get %s hardware address", ifreq.ifr_name);
- }
- memcpy (sockaddr.sll_addr, ifreq.ifr_ifru.ifru_hwaddr.sa_data, sizeof (sockaddr.sll_addr));
- if (bind (fd, (struct sockaddr *) (& sockaddr), sizeof (struct sockaddr_ll)) == - 1)
- {
- error (1, errno, "Can't bind socket to %s", ifreq.ifr_name);
- }
- while (! done)
- {
- signed length = recvfrom (fd, packet, sizeof (packet), 0, (struct sockaddr *) (0), (socklen_t *) (0));
- if (length > 0)
- {
- if (_allclr (flags, HPAVD_SILENCE))
- {
- MMEPeek (& packet, length, stdout);
- }
- if (_anyset (flags, HPAVD_VERBOSE))
- {
- hexdump (& packet, 0, length, stdout);
- }
- }
- }
- ifreq.ifr_flags = state;
- if (ioctl (fd, SIOCSIFFLAGS, & ifreq) < 0)
- {
- error (1, errno, "Can't restore %s device state", ifreq.ifr_name);
- }
- close (fd);
- return (0);
- }
|