Brian Silverman | f6b6884 | 2013-12-20 12:34:58 -0800 | [diff] [blame] | 1 | #include "bbb/uart_reader.h" |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 2 | |
Daniel Petti | a4fe7bc | 2013-12-22 12:57:50 -0800 | [diff] [blame] | 3 | #include <errno.h> |
Daniel Petti | b36f5b8 | 2013-12-15 08:55:04 -0800 | [diff] [blame] | 4 | #include <fcntl.h> |
| 5 | #include <linux/serial.h> |
Daniel Petti | b36f5b8 | 2013-12-15 08:55:04 -0800 | [diff] [blame] | 6 | #include <unistd.h> |
Daniel Petti | 059be42 | 2013-12-14 19:47:42 -0800 | [diff] [blame] | 7 | |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 8 | #include "aos/common/logging/logging.h" |
Brian Silverman | 53f2918 | 2013-12-21 15:16:27 -0800 | [diff] [blame] | 9 | |
Daniel Petti | 059be42 | 2013-12-14 19:47:42 -0800 | [diff] [blame] | 10 | // This is the code for receiving data from the cape via UART. |
Brian Silverman | 04fac62 | 2014-01-26 18:32:15 -0800 | [diff] [blame] | 11 | // fragment active. |
| 12 | // `su -c "echo BB-UART1 > /sys/devices/bone_capemgr.*/slots"` works, but |
| 13 | // you have to do it every time. It is also possible to set it up to do that |
| 14 | // every time it boots. |
Daniel Petti | b36f5b8 | 2013-12-15 08:55:04 -0800 | [diff] [blame] | 15 | |
Daniel Petti | 059be42 | 2013-12-14 19:47:42 -0800 | [diff] [blame] | 16 | namespace bbb { |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 17 | namespace { |
Brian Silverman | 0b88252 | 2014-02-16 15:47:34 -0800 | [diff] [blame] | 18 | |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 19 | // TODO(brians): Determine this in some way that allows easy switching for |
| 20 | // testing with /dev/ttyUSB0 for example. |
Brian Silverman | 0b6d9f4 | 2013-12-23 22:03:57 -0800 | [diff] [blame] | 21 | const char *device = "/dev/ttyO1"; |
Brian Silverman | 0b88252 | 2014-02-16 15:47:34 -0800 | [diff] [blame] | 22 | |
| 23 | bool easy_access(const char *path) { |
| 24 | if (access(path, R_OK | W_OK) == 0) return true; |
| 25 | if (errno == EACCES || errno == ENOENT) return false; |
| 26 | LOG(FATAL, "access(%s, F_OK) failed with %d: %s\n", path, errno, |
| 27 | strerror(errno)); |
| 28 | } |
| 29 | |
| 30 | int open_device() { |
| 31 | if (easy_access(device)) { |
| 32 | LOG(INFO, "unexporting BB-UART1\n"); |
| 33 | if (system("bash -c 'echo -$(cat /sys/devices/bone_capemgr.*/slots" |
| 34 | " | fgrep BB-UART1" |
Brian Silverman | 2508c08 | 2014-02-17 15:45:02 -0800 | [diff] [blame] | 35 | " | cut -d : -f 1 | tr -d \" \")" |
| 36 | " > /sys/devices/bone_capemgr.*/slots'") == -1) { |
Brian Silverman | 0b88252 | 2014-02-16 15:47:34 -0800 | [diff] [blame] | 37 | LOG(FATAL, "system([disable OMAP UART]) failed with %d: %s\n", errno, |
| 38 | strerror(errno)); |
| 39 | } |
| 40 | while (easy_access(device)) { |
Brian Silverman | 2508c08 | 2014-02-17 15:45:02 -0800 | [diff] [blame] | 41 | LOG(DEBUG, "waiting for BB-UART1 to be unexported\n"); |
Brian Silverman | 0b88252 | 2014-02-16 15:47:34 -0800 | [diff] [blame] | 42 | ::aos::time::SleepFor(::aos::time::Time::InSeconds(0.1)); |
| 43 | } |
| 44 | } |
| 45 | |
| 46 | LOG(INFO, "exporting BB-UART1\n"); |
| 47 | // 2 strings to work around a VIM bug where the indenter locks up when they're |
| 48 | // combined as 1... |
| 49 | if (system("bash -c 'echo BB-UART1 > /sys/devices/bone_capemgr.*" |
| 50 | "/slots'") == |
| 51 | -1) { |
| 52 | LOG(FATAL, "system([enable OMAP UART]) failed with %d: %s\n", errno, |
| 53 | strerror(errno)); |
| 54 | } |
| 55 | while (!easy_access(device)) { |
Brian Silverman | 2508c08 | 2014-02-17 15:45:02 -0800 | [diff] [blame] | 56 | LOG(DEBUG, "waiting for BB-UART1 to be exported\n"); |
Brian Silverman | 0b88252 | 2014-02-16 15:47:34 -0800 | [diff] [blame] | 57 | ::aos::time::SleepFor(::aos::time::Time::InSeconds(0.1)); |
| 58 | } |
| 59 | |
| 60 | return open(device, O_RDWR | O_NOCTTY | O_NONBLOCK); |
| 61 | } |
| 62 | |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 63 | } // namespace |
Daniel Petti | 059be42 | 2013-12-14 19:47:42 -0800 | [diff] [blame] | 64 | |
Brian Silverman | 0b6d9f4 | 2013-12-23 22:03:57 -0800 | [diff] [blame] | 65 | extern "C" { |
| 66 | |
| 67 | // Sets all of the weird, annoying TTY flags on fd. In a separate file because |
| 68 | // the structure it uses is so weird and annoying and carries so much baggage it |
| 69 | // messes up all the #includes in whatever file it's used in. |
Brian Silverman | c82c512 | 2013-12-25 15:53:44 -0800 | [diff] [blame] | 70 | // Defined in uart_reader_termios2.c. |
Brian Silverman | 0b6d9f4 | 2013-12-23 22:03:57 -0800 | [diff] [blame] | 71 | extern int aos_uart_reader_set_tty_options(int fd, int baud_rate); |
| 72 | |
| 73 | } // extern "C" |
| 74 | |
Brian Silverman | f6b6884 | 2013-12-20 12:34:58 -0800 | [diff] [blame] | 75 | UartReader::UartReader(int32_t baud_rate) |
Brian Silverman | 0b88252 | 2014-02-16 15:47:34 -0800 | [diff] [blame] | 76 | : fd_(open_device()) { |
Daniel Petti | a4fe7bc | 2013-12-22 12:57:50 -0800 | [diff] [blame] | 77 | |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 78 | if (fd_ < 0) { |
Brian Silverman | 50a9d03 | 2014-02-16 17:20:57 -0800 | [diff] [blame] | 79 | LOG(FATAL, "open(%s, O_RDWR | O_NOCTTY | O_NOBLOCK) failed with %d: %s\n", |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 80 | device, errno, strerror(errno)); |
Daniel Petti | 059be42 | 2013-12-14 19:47:42 -0800 | [diff] [blame] | 81 | } |
Brian Silverman | ed03006 | 2013-12-20 21:03:47 -0800 | [diff] [blame] | 82 | |
Brian Silverman | 0b6d9f4 | 2013-12-23 22:03:57 -0800 | [diff] [blame] | 83 | if (aos_uart_reader_set_tty_options(fd_, baud_rate) != 0) { |
| 84 | LOG(FATAL, "aos_uart_reader_set_tty_options(%d) failed with %d: %s\n", |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 85 | fd_, errno, strerror(errno)); |
| 86 | } |
Daniel Petti | 059be42 | 2013-12-14 19:47:42 -0800 | [diff] [blame] | 87 | } |
| 88 | |
Brian Silverman | f6b6884 | 2013-12-20 12:34:58 -0800 | [diff] [blame] | 89 | UartReader::~UartReader() { |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 90 | if (fd_ > 0) close(fd_); |
| 91 | } |
Daniel Petti | 059be42 | 2013-12-14 19:47:42 -0800 | [diff] [blame] | 92 | |
Brian Silverman | 04fac62 | 2014-01-26 18:32:15 -0800 | [diff] [blame] | 93 | ssize_t UartReader::ReadBytes(uint8_t *dest, size_t max_bytes, |
Brian Silverman | 7d16c57 | 2014-01-03 20:27:57 -0800 | [diff] [blame] | 94 | const ::aos::time::Time &timeout_time) { |
Brian Silverman | 803b7a5 | 2014-01-01 13:21:18 -0800 | [diff] [blame] | 95 | do { |
Brian Silverman | 7d16c57 | 2014-01-03 20:27:57 -0800 | [diff] [blame] | 96 | ::aos::time::Time timeout = timeout_time - ::aos::time::Time::Now(); |
| 97 | if (timeout < ::aos::time::Time(0, 0)) return -2; |
Brian Silverman | ff1218b | 2014-01-03 14:47:39 -0800 | [diff] [blame] | 98 | struct timeval timeout_timeval = timeout.ToTimeval(); |
Brian Silverman | 0fc4ca7 | 2014-03-02 17:26:06 -0800 | [diff] [blame^] | 99 | fd_set fds; |
| 100 | FD_ZERO(&fds); |
| 101 | FD_SET(fd_, &fds); |
| 102 | switch (select(fd_ + 1, &fds, NULL, NULL, &timeout_timeval)) { |
Brian Silverman | ff1218b | 2014-01-03 14:47:39 -0800 | [diff] [blame] | 103 | case 0: |
| 104 | return -2; |
| 105 | case -1: |
| 106 | continue; |
| 107 | case 1: |
| 108 | break; |
| 109 | default: |
| 110 | LOG(WARNING, "unknown select return value\n"); |
| 111 | return -1; |
| 112 | } |
Brian Silverman | 803b7a5 | 2014-01-01 13:21:18 -0800 | [diff] [blame] | 113 | ssize_t r = read(fd_, dest, max_bytes); |
| 114 | if (r != -1) return r; |
| 115 | } while (errno == EINTR); |
| 116 | return -1; |
Brian Silverman | 1662a0e | 2013-12-19 17:50:01 -0800 | [diff] [blame] | 117 | } |
Daniel Petti | 059be42 | 2013-12-14 19:47:42 -0800 | [diff] [blame] | 118 | |
Brian Silverman | 04fac62 | 2014-01-26 18:32:15 -0800 | [diff] [blame] | 119 | bool UartReader::WriteBytes(uint8_t *bytes, size_t number_bytes) { |
| 120 | size_t written = 0; |
| 121 | while (written < number_bytes) { |
| 122 | ssize_t r = write(fd_, &bytes[written], number_bytes - written); |
| 123 | if (r == -1) return false; |
| 124 | written += r; |
| 125 | } |
| 126 | return true; |
| 127 | } |
| 128 | |
Brian Silverman | ffeef3f | 2013-12-22 14:06:23 -0800 | [diff] [blame] | 129 | } // namespace bbb |