John Park | 398c74a | 2018-10-20 21:17:39 -0700 | [diff] [blame] | 1 | #include "aos/configuration.h" |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 2 | |
| 3 | #include <string.h> |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 4 | #include <stdlib.h> |
| 5 | #include <sys/types.h> |
| 6 | #include <netinet/in.h> |
| 7 | #include <arpa/inet.h> |
| 8 | #include <ifaddrs.h> |
| 9 | #include <unistd.h> |
| 10 | |
John Park | 33858a3 | 2018-09-28 23:05:48 -0700 | [diff] [blame] | 11 | #include "aos/logging/logging.h" |
| 12 | #include "aos/unique_malloc_ptr.h" |
Sabina Davis | 2ed5ea2 | 2017-09-26 22:27:42 -0700 | [diff] [blame] | 13 | #include "aos/once.h" |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 14 | |
| 15 | namespace aos { |
| 16 | namespace configuration { |
| 17 | namespace { |
| 18 | |
Austin Schuh | 629821e | 2014-02-10 21:18:27 -0800 | [diff] [blame] | 19 | // TODO(brians): This shouldn't be necesary for running tests. Provide a way to |
| 20 | // set the IP address when running tests from the test. |
Brian Silverman | 14fd0fb | 2014-01-14 21:42:01 -0800 | [diff] [blame] | 21 | const char *const kLinuxNetInterface = "eth0"; |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 22 | const in_addr *DoGetOwnIPAddress() { |
Brian Silverman | 01be000 | 2014-05-10 15:44:38 -0700 | [diff] [blame] | 23 | static const char *kOverrideVariable = "FRC971_IP_OVERRIDE"; |
| 24 | const char *override_ip = getenv(kOverrideVariable); |
| 25 | if (override_ip != NULL) { |
| 26 | LOG(INFO, "Override IP is %s\n", override_ip); |
| 27 | static in_addr r; |
| 28 | if (inet_aton(override_ip, &r) != 0) { |
| 29 | return &r; |
joe | 3779d0c | 2014-02-15 19:41:22 -0800 | [diff] [blame] | 30 | } else { |
Brian Silverman | 01be000 | 2014-05-10 15:44:38 -0700 | [diff] [blame] | 31 | LOG(WARNING, "error parsing %s value '%s'\n", |
| 32 | kOverrideVariable, override_ip); |
joe | 3779d0c | 2014-02-15 19:41:22 -0800 | [diff] [blame] | 33 | } |
Brian Silverman | 01be000 | 2014-05-10 15:44:38 -0700 | [diff] [blame] | 34 | } else { |
| 35 | LOG(INFO, "Couldn't get environmental variable.\n"); |
| 36 | } |
joe | 3779d0c | 2014-02-15 19:41:22 -0800 | [diff] [blame] | 37 | |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 38 | ifaddrs *addrs; |
| 39 | if (getifaddrs(&addrs) != 0) { |
Brian Silverman | 01be000 | 2014-05-10 15:44:38 -0700 | [diff] [blame] | 40 | PLOG(FATAL, "getifaddrs(%p) failed", &addrs); |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 41 | } |
| 42 | // Smart pointers don't work very well for iterating through a linked list, |
| 43 | // but it does do a very nice job of making sure that addrs gets freed. |
| 44 | unique_c_ptr<ifaddrs, freeifaddrs> addrs_deleter(addrs); |
| 45 | |
Austin Schuh | 629821e | 2014-02-10 21:18:27 -0800 | [diff] [blame] | 46 | for (; addrs != nullptr; addrs = addrs->ifa_next) { |
Brian Silverman | 6da0427 | 2014-05-18 18:47:48 -0700 | [diff] [blame] | 47 | // ifa_addr tends to be nullptr on CAN interfaces. |
Austin Schuh | 629821e | 2014-02-10 21:18:27 -0800 | [diff] [blame] | 48 | if (addrs->ifa_addr != nullptr && addrs->ifa_addr->sa_family == AF_INET) { |
Brian Silverman | 14fd0fb | 2014-01-14 21:42:01 -0800 | [diff] [blame] | 49 | if (strcmp(kLinuxNetInterface, addrs->ifa_name) == 0) { |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 50 | static const in_addr r = |
Brian Silverman | 63cf241 | 2013-11-17 05:44:36 -0800 | [diff] [blame] | 51 | reinterpret_cast<sockaddr_in *>(__builtin_assume_aligned( |
| 52 | addrs->ifa_addr, alignof(sockaddr_in)))->sin_addr; |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 53 | return &r; |
| 54 | } |
| 55 | } |
| 56 | } |
| 57 | LOG(FATAL, "couldn't find an AF_INET interface named \"%s\"\n", |
Brian Silverman | 14fd0fb | 2014-01-14 21:42:01 -0800 | [diff] [blame] | 58 | kLinuxNetInterface); |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 59 | } |
| 60 | |
| 61 | const char *DoGetRootDirectory() { |
| 62 | ssize_t size = 0; |
| 63 | char *r = NULL; |
| 64 | while (true) { |
| 65 | if (r != NULL) delete r; |
| 66 | size += 256; |
| 67 | r = new char[size]; |
| 68 | |
| 69 | ssize_t ret = readlink("/proc/self/exe", r, size); |
| 70 | if (ret < 0) { |
| 71 | if (ret != -1) { |
| 72 | LOG(WARNING, "it returned %zd, not -1\n", ret); |
| 73 | } |
Brian Silverman | 01be000 | 2014-05-10 15:44:38 -0700 | [diff] [blame] | 74 | PLOG(FATAL, "readlink(\"/proc/self/exe\", %p, %zu) failed", r, size); |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 75 | } |
| 76 | if (ret < size) { |
Brian Silverman | e6335e4 | 2014-02-20 20:53:06 -0800 | [diff] [blame] | 77 | void *last_slash = memrchr(r, '/', ret); |
Brian Silverman | 66f079a | 2013-08-26 16:24:30 -0700 | [diff] [blame] | 78 | if (last_slash == NULL) { |
| 79 | r[ret] = '\0'; |
| 80 | LOG(FATAL, "couldn't find a '/' in \"%s\"\n", r); |
| 81 | } |
| 82 | *static_cast<char *>(last_slash) = '\0'; |
| 83 | LOG(INFO, "got a root dir of \"%s\"\n", r); |
| 84 | return r; |
| 85 | } |
| 86 | } |
| 87 | } |
| 88 | |
| 89 | const char *DoGetLoggingDirectory() { |
| 90 | static const char kSuffix[] = "/../../tmp/robot_logs"; |
| 91 | const char *root = GetRootDirectory(); |
| 92 | char *r = new char[strlen(root) + sizeof(kSuffix)]; |
| 93 | strcpy(r, root); |
| 94 | strcat(r, kSuffix); |
| 95 | return r; |
| 96 | } |
| 97 | |
| 98 | } // namespace |
| 99 | |
| 100 | const char *GetRootDirectory() { |
| 101 | static aos::Once<const char> once(DoGetRootDirectory); |
| 102 | return once.Get(); |
| 103 | } |
| 104 | |
| 105 | const char *GetLoggingDirectory() { |
| 106 | static aos::Once<const char> once(DoGetLoggingDirectory); |
| 107 | return once.Get(); |
| 108 | } |
| 109 | |
| 110 | const in_addr &GetOwnIPAddress() { |
| 111 | static aos::Once<const in_addr> once(DoGetOwnIPAddress); |
| 112 | return *once.Get(); |
| 113 | } |
| 114 | |
| 115 | } // namespace configuration |
| 116 | } // namespace aos |