| /* |
| * |
| * BlueZ - Bluetooth protocol stack for Linux |
| * |
| * Copyright (C) 2002-2010 Marcel Holtmann <marcel@holtmann.org> |
| * |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA |
| * |
| */ |
| |
| #ifdef HAVE_CONFIG_H |
| #include <config.h> |
| #endif |
| |
| #include <stdio.h> |
| #include <errno.h> |
| #include <fcntl.h> |
| #include <unistd.h> |
| #include <stdlib.h> |
| #include <signal.h> |
| #include <syslog.h> |
| #include <getopt.h> |
| #include <sys/poll.h> |
| #include <sys/ioctl.h> |
| #include <sys/socket.h> |
| |
| #include <bluetooth/bluetooth.h> |
| #include <bluetooth/rfcomm.h> |
| |
| /* IO cancelation */ |
| static volatile sig_atomic_t __io_canceled; |
| |
| static inline void io_init(void) |
| { |
| __io_canceled = 0; |
| } |
| |
| static inline void io_cancel(void) |
| { |
| __io_canceled = 1; |
| } |
| |
| /* Signal functions */ |
| static void sig_hup(int sig) |
| { |
| return; |
| } |
| |
| static void sig_term(int sig) |
| { |
| syslog(LOG_INFO, "Closing RFCOMM channel"); |
| io_cancel(); |
| } |
| |
| /* Read exactly len bytes (Signal safe)*/ |
| static inline int read_n(int fd, char *buf, int len) |
| { |
| register int t = 0, w; |
| |
| while (!__io_canceled && len > 0) { |
| if ((w = read(fd, buf, len)) < 0) { |
| if (errno == EINTR || errno == EAGAIN) |
| continue; |
| return -1; |
| } |
| if (!w) |
| return 0; |
| len -= w; |
| buf += w; |
| t += w; |
| } |
| |
| return t; |
| } |
| |
| /* Write exactly len bytes (Signal safe)*/ |
| static inline int write_n(int fd, char *buf, int len) |
| { |
| register int t = 0, w; |
| |
| while (!__io_canceled && len > 0) { |
| if ((w = write(fd, buf, len)) < 0) { |
| if (errno == EINTR || errno == EAGAIN) |
| continue; |
| return -1; |
| } |
| if (!w) |
| return 0; |
| len -= w; |
| buf += w; |
| t += w; |
| } |
| |
| return t; |
| } |
| |
| /* Create the RFCOMM connection */ |
| static int create_connection(bdaddr_t *bdaddr, uint8_t channel) |
| { |
| struct sockaddr_rc remote_addr, local_addr; |
| int fd, err; |
| |
| if ((fd = socket(PF_BLUETOOTH, SOCK_STREAM, BTPROTO_RFCOMM)) < 0) |
| return fd; |
| |
| memset(&local_addr, 0, sizeof(local_addr)); |
| local_addr.rc_family = AF_BLUETOOTH; |
| bacpy(&local_addr.rc_bdaddr, BDADDR_ANY); |
| if ((err = bind(fd, (struct sockaddr *)&local_addr, sizeof(local_addr))) < 0) { |
| close(fd); |
| return err; |
| } |
| |
| memset(&remote_addr, 0, sizeof(remote_addr)); |
| remote_addr.rc_family = AF_BLUETOOTH; |
| bacpy(&remote_addr.rc_bdaddr, bdaddr); |
| remote_addr.rc_channel = channel; |
| if ((err = connect(fd, (struct sockaddr *)&remote_addr, sizeof(remote_addr))) < 0) { |
| close(fd); |
| return err; |
| } |
| |
| syslog(LOG_INFO, "RFCOMM channel %d connected", channel); |
| |
| return fd; |
| } |
| |
| /* Process the data from socket and pseudo tty */ |
| static int process_data(int fd) |
| { |
| struct pollfd p[2]; |
| char buf[1024]; |
| int err, r; |
| |
| p[0].fd = 0; |
| p[0].events = POLLIN | POLLERR | POLLHUP | POLLNVAL; |
| |
| p[1].fd = fd; |
| p[1].events = POLLIN | POLLERR | POLLHUP | POLLNVAL; |
| |
| err = 0; |
| |
| while (!__io_canceled) { |
| p[0].revents = 0; |
| p[1].revents = 0; |
| |
| err = poll(p, 2, -1); |
| if (err < 0) |
| break; |
| |
| err = 0; |
| |
| if (p[0].revents) { |
| if (p[0].revents & (POLLERR | POLLHUP | POLLNVAL)) |
| break; |
| r = read(0, buf, sizeof(buf)); |
| if (r < 0) { |
| if (errno != EINTR && errno != EAGAIN) { |
| err = r; |
| break; |
| } |
| } |
| |
| err = write_n(fd, buf, r); |
| if (err < 0) |
| break; |
| } |
| |
| if (p[1].revents) { |
| if (p[1].revents & (POLLERR | POLLHUP | POLLNVAL)) |
| break; |
| r = read(fd, buf, sizeof(buf)); |
| if (r < 0) { |
| if (errno != EINTR && errno != EAGAIN) { |
| err = r; |
| break; |
| } |
| } |
| |
| err = write_n(1, buf, r); |
| if (err < 0) |
| break; |
| } |
| } |
| |
| return err; |
| } |
| |
| static void usage(void) |
| { |
| printf("Usage:\tppporc <bdaddr> [channel]\n"); |
| } |
| |
| int main(int argc, char** argv) |
| { |
| struct sigaction sa; |
| int fd, err, opt; |
| |
| bdaddr_t bdaddr; |
| uint8_t channel; |
| |
| /* Parse command line options */ |
| while ((opt = getopt(argc, argv, "h")) != EOF) { |
| switch(opt) { |
| case 'h': |
| usage(); |
| exit(0); |
| } |
| } |
| |
| argc -= optind; |
| argv += optind; |
| |
| switch (argc) { |
| case 1: |
| str2ba(argv[0], &bdaddr); |
| channel = 1; |
| break; |
| case 2: |
| str2ba(argv[0], &bdaddr); |
| channel = atoi(argv[1]); |
| break; |
| default: |
| usage(); |
| exit(0); |
| } |
| |
| /* Initialize syslog */ |
| openlog("ppporc", LOG_PID | LOG_NDELAY | LOG_PERROR, LOG_DAEMON); |
| syslog(LOG_INFO, "PPP over RFCOMM"); |
| |
| /* Initialize signals */ |
| memset(&sa, 0, sizeof(sa)); |
| sa.sa_flags = SA_NOCLDSTOP; |
| sa.sa_handler = SIG_IGN; |
| sigaction(SIGCHLD, &sa, NULL); |
| sigaction(SIGPIPE, &sa, NULL); |
| |
| sa.sa_handler = sig_term; |
| sigaction(SIGTERM, &sa, NULL); |
| sigaction(SIGINT, &sa, NULL); |
| |
| sa.sa_handler = sig_hup; |
| sigaction(SIGHUP, &sa, NULL); |
| |
| syslog(LOG_INFO, "Connecting to %s", argv[0]); |
| |
| if ((fd = create_connection(&bdaddr, channel)) < 0) { |
| syslog(LOG_ERR, "Can't connect to remote device (%s)", strerror(errno)); |
| return fd; |
| } |
| |
| err = process_data(fd); |
| |
| close(fd); |
| |
| return err; |
| } |