mirror of
https://git.kernel.org/pub/scm/bluetooth/bluez.git
synced 2024-11-24 20:54:19 +08:00
8a0285383c
The timeout when waiting for the RFCOMM tty to show up is intended to be 100ms and not 100us (rfcomm connect has this too).
846 lines
17 KiB
C
846 lines
17 KiB
C
/*
|
|
*
|
|
* BlueZ - Bluetooth protocol stack for Linux
|
|
*
|
|
* Copyright (C) 2002-2009 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
|
|
|
|
#define _GNU_SOURCE
|
|
#include <stdio.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <getopt.h>
|
|
#include <signal.h>
|
|
#include <termios.h>
|
|
#include <sys/poll.h>
|
|
#include <sys/param.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/socket.h>
|
|
#include <sys/wait.h>
|
|
|
|
#include <bluetooth/bluetooth.h>
|
|
#include <bluetooth/hci.h>
|
|
#include <bluetooth/hci_lib.h>
|
|
#include <bluetooth/rfcomm.h>
|
|
|
|
#include "kword.h"
|
|
|
|
#ifdef NEED_PPOLL
|
|
#include "ppoll.h"
|
|
#endif
|
|
|
|
static char *rfcomm_config_file = NULL;
|
|
static int rfcomm_raw_tty = 0;
|
|
static int auth = 0;
|
|
static int encryption = 0;
|
|
static int secure = 0;
|
|
static int master = 0;
|
|
static int linger = 0;
|
|
|
|
static char *rfcomm_state[] = {
|
|
"unknown",
|
|
"connected",
|
|
"clean",
|
|
"bound",
|
|
"listening",
|
|
"connecting",
|
|
"connecting",
|
|
"config",
|
|
"disconnecting",
|
|
"closed"
|
|
};
|
|
|
|
static volatile sig_atomic_t __io_canceled = 0;
|
|
|
|
static void sig_hup(int sig)
|
|
{
|
|
return;
|
|
}
|
|
|
|
static void sig_term(int sig)
|
|
{
|
|
__io_canceled = 1;
|
|
}
|
|
|
|
static char *rfcomm_flagstostr(uint32_t flags)
|
|
{
|
|
static char str[100];
|
|
str[0] = 0;
|
|
|
|
strcat(str, "[");
|
|
|
|
if (flags & (1 << RFCOMM_REUSE_DLC))
|
|
strcat(str, "reuse-dlc ");
|
|
|
|
if (flags & (1 << RFCOMM_RELEASE_ONHUP))
|
|
strcat(str, "release-on-hup ");
|
|
|
|
if (flags & (1 << RFCOMM_TTY_ATTACHED))
|
|
strcat(str, "tty-attached");
|
|
|
|
strcat(str, "]");
|
|
return str;
|
|
}
|
|
|
|
static void print_dev_info(struct rfcomm_dev_info *di)
|
|
{
|
|
char src[18], dst[18], addr[40];
|
|
|
|
ba2str(&di->src, src); ba2str(&di->dst, dst);
|
|
|
|
if (bacmp(&di->src, BDADDR_ANY) == 0)
|
|
sprintf(addr, "%s", dst);
|
|
else
|
|
sprintf(addr, "%s -> %s", src, dst);
|
|
|
|
printf("rfcomm%d: %s channel %d %s %s\n",
|
|
di->id, addr, di->channel,
|
|
rfcomm_state[di->state],
|
|
di->flags ? rfcomm_flagstostr(di->flags) : "");
|
|
}
|
|
|
|
static void print_dev_list(int ctl, int flags)
|
|
{
|
|
struct rfcomm_dev_list_req *dl;
|
|
struct rfcomm_dev_info *di;
|
|
int i;
|
|
|
|
dl = malloc(sizeof(*dl) + RFCOMM_MAX_DEV * sizeof(*di));
|
|
if (!dl) {
|
|
perror("Can't allocate memory");
|
|
exit(1);
|
|
}
|
|
|
|
dl->dev_num = RFCOMM_MAX_DEV;
|
|
di = dl->dev_info;
|
|
|
|
if (ioctl(ctl, RFCOMMGETDEVLIST, (void *) dl) < 0) {
|
|
perror("Can't get device list");
|
|
exit(1);
|
|
}
|
|
|
|
for (i = 0; i < dl->dev_num; i++)
|
|
print_dev_info(di + i);
|
|
}
|
|
|
|
static int create_dev(int ctl, int dev, uint32_t flags, bdaddr_t *bdaddr, int argc, char **argv)
|
|
{
|
|
struct rfcomm_dev_req req;
|
|
int err;
|
|
|
|
memset(&req, 0, sizeof(req));
|
|
req.dev_id = dev;
|
|
req.flags = flags;
|
|
bacpy(&req.src, bdaddr);
|
|
|
|
if (argc < 2) {
|
|
err = rfcomm_read_config(rfcomm_config_file);
|
|
if (err < 0) {
|
|
perror("Can't open RFCOMM config file");
|
|
return err;
|
|
}
|
|
|
|
bacpy(&req.dst, &rfcomm_opts[dev].bdaddr);
|
|
req.channel = rfcomm_opts[dev].channel;
|
|
|
|
if (bacmp(&req.dst, BDADDR_ANY) == 0) {
|
|
fprintf(stderr, "Can't find a config entry for rfcomm%d\n", dev);
|
|
return -EFAULT;
|
|
}
|
|
} else {
|
|
str2ba(argv[1], &req.dst);
|
|
|
|
if (argc > 2)
|
|
req.channel = atoi(argv[2]);
|
|
else
|
|
req.channel = 1;
|
|
}
|
|
|
|
err = ioctl(ctl, RFCOMMCREATEDEV, &req);
|
|
if (err == EOPNOTSUPP)
|
|
fprintf(stderr, "RFCOMM TTY support not available\n");
|
|
else if (err < 0)
|
|
perror("Can't create device");
|
|
|
|
return err;
|
|
}
|
|
|
|
static int create_all(int ctl)
|
|
{
|
|
struct rfcomm_dev_req req;
|
|
int i, err;
|
|
|
|
err = rfcomm_read_config(rfcomm_config_file);
|
|
if (err < 0) {
|
|
perror("Can't open RFCOMM config file");
|
|
return err;
|
|
}
|
|
|
|
for (i = 0; i < RFCOMM_MAX_DEV; i++) {
|
|
if (!rfcomm_opts[i].bind)
|
|
continue;
|
|
|
|
memset(&req, 0, sizeof(req));
|
|
req.dev_id = i;
|
|
req.flags = 0;
|
|
bacpy(&req.src, BDADDR_ANY);
|
|
bacpy(&req.dst, &rfcomm_opts[i].bdaddr);
|
|
req.channel = rfcomm_opts[i].channel;
|
|
|
|
if (bacmp(&req.dst, BDADDR_ANY) != 0)
|
|
ioctl(ctl, RFCOMMCREATEDEV, &req);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int release_dev(int ctl, int dev, uint32_t flags)
|
|
{
|
|
struct rfcomm_dev_req req;
|
|
int err;
|
|
|
|
memset(&req, 0, sizeof(req));
|
|
req.dev_id = dev;
|
|
|
|
err = ioctl(ctl, RFCOMMRELEASEDEV, &req);
|
|
if (err < 0)
|
|
perror("Can't release device");
|
|
|
|
return err;
|
|
}
|
|
|
|
static int release_all(int ctl)
|
|
{
|
|
struct rfcomm_dev_list_req *dl;
|
|
struct rfcomm_dev_info *di;
|
|
int i;
|
|
|
|
dl = malloc(sizeof(*dl) + RFCOMM_MAX_DEV * sizeof(*di));
|
|
if (!dl) {
|
|
perror("Can't allocate memory");
|
|
exit(1);
|
|
}
|
|
|
|
dl->dev_num = RFCOMM_MAX_DEV;
|
|
di = dl->dev_info;
|
|
|
|
if (ioctl(ctl, RFCOMMGETDEVLIST, (void *) dl) < 0) {
|
|
perror("Can't get device list");
|
|
exit(1);
|
|
}
|
|
|
|
for (i = 0; i < dl->dev_num; i++)
|
|
release_dev(ctl, (di + i)->id, 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void run_cmdline(struct pollfd *p, sigset_t* sigs, char *devname,
|
|
int argc, char **argv)
|
|
{
|
|
int i;
|
|
pid_t pid;
|
|
char **cmdargv;
|
|
|
|
cmdargv = malloc((argc + 1) * sizeof(char*));
|
|
if (!cmdargv)
|
|
return;
|
|
|
|
for (i = 0; i < argc; i++)
|
|
cmdargv[i] = (strcmp(argv[i], "{}") == 0) ? devname : argv[i];
|
|
cmdargv[i] = NULL;
|
|
|
|
pid = fork();
|
|
|
|
switch (pid) {
|
|
case 0:
|
|
i = execvp(cmdargv[0], cmdargv);
|
|
fprintf(stderr, "Couldn't execute command %s (errno=%d:%s)\n",
|
|
cmdargv[0], errno, strerror(errno));
|
|
break;
|
|
case -1:
|
|
fprintf(stderr, "Couldn't fork to execute command %s\n",
|
|
cmdargv[0]);
|
|
break;
|
|
default:
|
|
while (1) {
|
|
int status;
|
|
pid_t child;
|
|
struct timespec ts;
|
|
|
|
child = waitpid(-1, &status, WNOHANG);
|
|
if (child == pid || (child < 0 && errno != EAGAIN))
|
|
break;
|
|
|
|
p->revents = 0;
|
|
ts.tv_sec = 0;
|
|
ts.tv_nsec = 200;
|
|
if (ppoll(p, 1, &ts, sigs) || __io_canceled) {
|
|
kill(pid, SIGTERM);
|
|
waitpid(pid, &status, 0);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
free(cmdargv);
|
|
}
|
|
|
|
static void cmd_connect(int ctl, int dev, bdaddr_t *bdaddr, int argc, char **argv)
|
|
{
|
|
struct sockaddr_rc laddr, raddr;
|
|
struct rfcomm_dev_req req;
|
|
struct termios ti;
|
|
struct sigaction sa;
|
|
struct pollfd p;
|
|
sigset_t sigs;
|
|
socklen_t alen;
|
|
char dst[18], devname[MAXPATHLEN];
|
|
int sk, fd, try = 30;
|
|
|
|
laddr.rc_family = AF_BLUETOOTH;
|
|
bacpy(&laddr.rc_bdaddr, bdaddr);
|
|
laddr.rc_channel = 0;
|
|
|
|
if (argc < 2) {
|
|
if (rfcomm_read_config(rfcomm_config_file) < 0) {
|
|
perror("Can't open RFCOMM config file");
|
|
return;
|
|
}
|
|
|
|
raddr.rc_family = AF_BLUETOOTH;
|
|
bacpy(&raddr.rc_bdaddr, &rfcomm_opts[dev].bdaddr);
|
|
raddr.rc_channel = rfcomm_opts[dev].channel;
|
|
|
|
if (bacmp(&raddr.rc_bdaddr, BDADDR_ANY) == 0) {
|
|
fprintf(stderr, "Can't find a config entry for rfcomm%d\n", dev);
|
|
return;
|
|
}
|
|
} else {
|
|
raddr.rc_family = AF_BLUETOOTH;
|
|
str2ba(argv[1], &raddr.rc_bdaddr);
|
|
|
|
if (argc > 2)
|
|
raddr.rc_channel = atoi(argv[2]);
|
|
else
|
|
raddr.rc_channel = 1;
|
|
}
|
|
|
|
sk = socket(AF_BLUETOOTH, SOCK_STREAM, BTPROTO_RFCOMM);
|
|
if (sk < 0) {
|
|
perror("Can't create RFCOMM socket");
|
|
return;
|
|
}
|
|
|
|
if (linger) {
|
|
struct linger l = { .l_onoff = 1, .l_linger = linger };
|
|
|
|
if (setsockopt(sk, SOL_SOCKET, SO_LINGER, &l, sizeof(l)) < 0) {
|
|
perror("Can't set linger option");
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (bind(sk, (struct sockaddr *) &laddr, sizeof(laddr)) < 0) {
|
|
perror("Can't bind RFCOMM socket");
|
|
close(sk);
|
|
return;
|
|
}
|
|
|
|
if (connect(sk, (struct sockaddr *) &raddr, sizeof(raddr)) < 0) {
|
|
perror("Can't connect RFCOMM socket");
|
|
close(sk);
|
|
return;
|
|
}
|
|
|
|
alen = sizeof(laddr);
|
|
if (getsockname(sk, (struct sockaddr *)&laddr, &alen) < 0) {
|
|
perror("Can't get RFCOMM socket name");
|
|
close(sk);
|
|
return;
|
|
}
|
|
|
|
memset(&req, 0, sizeof(req));
|
|
req.dev_id = dev;
|
|
req.flags = (1 << RFCOMM_REUSE_DLC) | (1 << RFCOMM_RELEASE_ONHUP);
|
|
|
|
bacpy(&req.src, &laddr.rc_bdaddr);
|
|
bacpy(&req.dst, &raddr.rc_bdaddr);
|
|
req.channel = raddr.rc_channel;
|
|
|
|
dev = ioctl(sk, RFCOMMCREATEDEV, &req);
|
|
if (dev < 0) {
|
|
perror("Can't create RFCOMM TTY");
|
|
close(sk);
|
|
return;
|
|
}
|
|
|
|
snprintf(devname, MAXPATHLEN - 1, "/dev/rfcomm%d", dev);
|
|
while ((fd = open(devname, O_RDONLY | O_NOCTTY)) < 0) {
|
|
if (errno == EACCES) {
|
|
perror("Can't open RFCOMM device");
|
|
goto release;
|
|
}
|
|
|
|
snprintf(devname, MAXPATHLEN - 1, "/dev/bluetooth/rfcomm/%d", dev);
|
|
if ((fd = open(devname, O_RDONLY | O_NOCTTY)) < 0) {
|
|
if (try--) {
|
|
snprintf(devname, MAXPATHLEN - 1, "/dev/rfcomm%d", dev);
|
|
usleep(100 * 1000);
|
|
continue;
|
|
}
|
|
perror("Can't open RFCOMM device");
|
|
goto release;
|
|
}
|
|
}
|
|
|
|
if (rfcomm_raw_tty) {
|
|
tcflush(fd, TCIOFLUSH);
|
|
|
|
cfmakeraw(&ti);
|
|
tcsetattr(fd, TCSANOW, &ti);
|
|
}
|
|
|
|
close(sk);
|
|
|
|
ba2str(&req.dst, dst);
|
|
printf("Connected %s to %s on channel %d\n", devname, dst, req.channel);
|
|
printf("Press CTRL-C for hangup\n");
|
|
|
|
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);
|
|
|
|
sigfillset(&sigs);
|
|
sigdelset(&sigs, SIGCHLD);
|
|
sigdelset(&sigs, SIGPIPE);
|
|
sigdelset(&sigs, SIGTERM);
|
|
sigdelset(&sigs, SIGINT);
|
|
sigdelset(&sigs, SIGHUP);
|
|
|
|
p.fd = fd;
|
|
p.events = POLLERR | POLLHUP;
|
|
|
|
while (!__io_canceled) {
|
|
p.revents = 0;
|
|
if (ppoll(&p, 1, NULL, &sigs) > 0)
|
|
break;
|
|
}
|
|
|
|
printf("Disconnected\n");
|
|
|
|
close(fd);
|
|
return;
|
|
|
|
release:
|
|
memset(&req, 0, sizeof(req));
|
|
req.dev_id = dev;
|
|
req.flags = (1 << RFCOMM_HANGUP_NOW);
|
|
ioctl(ctl, RFCOMMRELEASEDEV, &req);
|
|
|
|
close(sk);
|
|
}
|
|
|
|
static void cmd_listen(int ctl, int dev, bdaddr_t *bdaddr, int argc, char **argv)
|
|
{
|
|
struct sockaddr_rc laddr, raddr;
|
|
struct rfcomm_dev_req req;
|
|
struct termios ti;
|
|
struct sigaction sa;
|
|
struct pollfd p;
|
|
sigset_t sigs;
|
|
socklen_t alen;
|
|
char dst[18], devname[MAXPATHLEN];
|
|
int sk, nsk, fd, lm, try = 30;
|
|
|
|
laddr.rc_family = AF_BLUETOOTH;
|
|
bacpy(&laddr.rc_bdaddr, bdaddr);
|
|
laddr.rc_channel = (argc < 2) ? 1 : atoi(argv[1]);
|
|
|
|
sk = socket(AF_BLUETOOTH, SOCK_STREAM, BTPROTO_RFCOMM);
|
|
if (sk < 0) {
|
|
perror("Can't create RFCOMM socket");
|
|
return;
|
|
}
|
|
|
|
lm = 0;
|
|
if (master)
|
|
lm |= RFCOMM_LM_MASTER;
|
|
if (auth)
|
|
lm |= RFCOMM_LM_AUTH;
|
|
if (encryption)
|
|
lm |= RFCOMM_LM_ENCRYPT;
|
|
if (secure)
|
|
lm |= RFCOMM_LM_SECURE;
|
|
|
|
if (lm && setsockopt(sk, SOL_RFCOMM, RFCOMM_LM, &lm, sizeof(lm)) < 0) {
|
|
perror("Can't set RFCOMM link mode");
|
|
close(sk);
|
|
return;
|
|
}
|
|
|
|
if (bind(sk, (struct sockaddr *)&laddr, sizeof(laddr)) < 0) {
|
|
perror("Can't bind RFCOMM socket");
|
|
close(sk);
|
|
return;
|
|
}
|
|
|
|
printf("Waiting for connection on channel %d\n", laddr.rc_channel);
|
|
|
|
listen(sk, 10);
|
|
|
|
alen = sizeof(raddr);
|
|
nsk = accept(sk, (struct sockaddr *) &raddr, &alen);
|
|
|
|
alen = sizeof(laddr);
|
|
if (getsockname(nsk, (struct sockaddr *)&laddr, &alen) < 0) {
|
|
perror("Can't get RFCOMM socket name");
|
|
close(nsk);
|
|
return;
|
|
}
|
|
|
|
if (linger) {
|
|
struct linger l = { .l_onoff = 1, .l_linger = linger };
|
|
|
|
if (setsockopt(nsk, SOL_SOCKET, SO_LINGER, &l, sizeof(l)) < 0) {
|
|
perror("Can't set linger option");
|
|
close(nsk);
|
|
return;
|
|
}
|
|
}
|
|
|
|
memset(&req, 0, sizeof(req));
|
|
req.dev_id = dev;
|
|
req.flags = (1 << RFCOMM_REUSE_DLC) | (1 << RFCOMM_RELEASE_ONHUP);
|
|
|
|
bacpy(&req.src, &laddr.rc_bdaddr);
|
|
bacpy(&req.dst, &raddr.rc_bdaddr);
|
|
req.channel = raddr.rc_channel;
|
|
|
|
dev = ioctl(nsk, RFCOMMCREATEDEV, &req);
|
|
if (dev < 0) {
|
|
perror("Can't create RFCOMM TTY");
|
|
close(sk);
|
|
return;
|
|
}
|
|
|
|
snprintf(devname, MAXPATHLEN - 1, "/dev/rfcomm%d", dev);
|
|
while ((fd = open(devname, O_RDONLY | O_NOCTTY)) < 0) {
|
|
if (errno == EACCES) {
|
|
perror("Can't open RFCOMM device");
|
|
goto release;
|
|
}
|
|
|
|
snprintf(devname, MAXPATHLEN - 1, "/dev/bluetooth/rfcomm/%d", dev);
|
|
if ((fd = open(devname, O_RDONLY | O_NOCTTY)) < 0) {
|
|
if (try--) {
|
|
snprintf(devname, MAXPATHLEN - 1, "/dev/rfcomm%d", dev);
|
|
usleep(100 * 1000);
|
|
continue;
|
|
}
|
|
perror("Can't open RFCOMM device");
|
|
goto release;
|
|
}
|
|
}
|
|
|
|
if (rfcomm_raw_tty) {
|
|
tcflush(fd, TCIOFLUSH);
|
|
|
|
cfmakeraw(&ti);
|
|
tcsetattr(fd, TCSANOW, &ti);
|
|
}
|
|
|
|
close(sk);
|
|
close(nsk);
|
|
|
|
ba2str(&req.dst, dst);
|
|
printf("Connection from %s to %s\n", dst, devname);
|
|
printf("Press CTRL-C for hangup\n");
|
|
|
|
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);
|
|
|
|
sigfillset(&sigs);
|
|
sigdelset(&sigs, SIGCHLD);
|
|
sigdelset(&sigs, SIGPIPE);
|
|
sigdelset(&sigs, SIGTERM);
|
|
sigdelset(&sigs, SIGINT);
|
|
sigdelset(&sigs, SIGHUP);
|
|
|
|
p.fd = fd;
|
|
p.events = POLLERR | POLLHUP;
|
|
|
|
if (argc <= 2) {
|
|
while (!__io_canceled) {
|
|
p.revents = 0;
|
|
if (ppoll(&p, 1, NULL, &sigs) > 0)
|
|
break;
|
|
}
|
|
} else
|
|
run_cmdline(&p, &sigs, devname, argc - 2, argv + 2);
|
|
|
|
sa.sa_handler = NULL;
|
|
sigaction(SIGTERM, &sa, NULL);
|
|
sigaction(SIGINT, &sa, NULL);
|
|
|
|
printf("Disconnected\n");
|
|
|
|
close(fd);
|
|
return;
|
|
|
|
release:
|
|
memset(&req, 0, sizeof(req));
|
|
req.dev_id = dev;
|
|
req.flags = (1 << RFCOMM_HANGUP_NOW);
|
|
ioctl(ctl, RFCOMMRELEASEDEV, &req);
|
|
|
|
close(sk);
|
|
}
|
|
|
|
static void cmd_watch(int ctl, int dev, bdaddr_t *bdaddr, int argc, char **argv)
|
|
{
|
|
while (!__io_canceled) {
|
|
cmd_listen(ctl, dev, bdaddr, argc, argv);
|
|
usleep(10000);
|
|
}
|
|
}
|
|
|
|
static void cmd_create(int ctl, int dev, bdaddr_t *bdaddr, int argc, char **argv)
|
|
{
|
|
if (strcmp(argv[0], "all") == 0)
|
|
create_all(ctl);
|
|
else
|
|
create_dev(ctl, dev, 0, bdaddr, argc, argv);
|
|
}
|
|
|
|
static void cmd_release(int ctl, int dev, bdaddr_t *bdaddr, int argc, char **argv)
|
|
{
|
|
if (strcmp(argv[0], "all") == 0)
|
|
release_all(ctl);
|
|
else
|
|
release_dev(ctl, dev, 0);
|
|
}
|
|
|
|
static void cmd_show(int ctl, int dev, bdaddr_t *bdaddr, int argc, char **argv)
|
|
{
|
|
if (strcmp(argv[0], "all") == 0)
|
|
print_dev_list(ctl, 0);
|
|
else {
|
|
struct rfcomm_dev_info di = { id: atoi(argv[0]) };
|
|
if (ioctl(ctl, RFCOMMGETDEVINFO, &di) < 0) {
|
|
perror("Get info failed");
|
|
exit(1);
|
|
}
|
|
|
|
print_dev_info(&di);
|
|
}
|
|
}
|
|
|
|
struct {
|
|
char *cmd;
|
|
char *alt;
|
|
void (*func)(int ctl, int dev, bdaddr_t *bdaddr, int argc, char **argv);
|
|
char *opt;
|
|
char *doc;
|
|
} command[] = {
|
|
{ "bind", "create", cmd_create, "<dev> <bdaddr> [channel]", "Bind device" },
|
|
{ "release", "unbind", cmd_release, "<dev>", "Release device" },
|
|
{ "show", "info", cmd_show, "<dev>", "Show device" },
|
|
{ "connect", "conn", cmd_connect, "<dev> <bdaddr> [channel]", "Connect device" },
|
|
{ "listen", "server", cmd_listen, "<dev> [channel [cmd]]", "Listen" },
|
|
{ "watch", "watch", cmd_watch, "<dev> [channel [cmd]]", "Watch" },
|
|
{ NULL, NULL, NULL, 0, 0 }
|
|
};
|
|
|
|
static void usage(void)
|
|
{
|
|
int i;
|
|
|
|
printf("RFCOMM configuration utility ver %s\n", VERSION);
|
|
|
|
printf("Usage:\n"
|
|
"\trfcomm [options] <command> <dev>\n"
|
|
"\n");
|
|
|
|
printf("Options:\n"
|
|
"\t-i [hciX|bdaddr] Local HCI device or BD Address\n"
|
|
"\t-h, --help Display help\n"
|
|
"\t-r, --raw Switch TTY into raw mode\n"
|
|
"\t-A, --auth Enable authentication\n"
|
|
"\t-E, --encrypt Enable encryption\n"
|
|
"\t-S, --secure Secure connection\n"
|
|
"\t-M, --master Become the master of a piconet\n"
|
|
"\t-f, --config [file] Specify alternate config file\n"
|
|
"\t-a Show all devices (default)\n"
|
|
"\n");
|
|
|
|
printf("Commands:\n");
|
|
for (i = 0; command[i].cmd; i++)
|
|
printf("\t%-8s %-24s\t%s\n",
|
|
command[i].cmd,
|
|
command[i].opt ? command[i].opt : " ",
|
|
command[i].doc);
|
|
printf("\n");
|
|
}
|
|
|
|
static struct option main_options[] = {
|
|
{ "help", 0, 0, 'h' },
|
|
{ "device", 1, 0, 'i' },
|
|
{ "config", 1, 0, 'f' },
|
|
{ "raw", 0, 0, 'r' },
|
|
{ "auth", 0, 0, 'A' },
|
|
{ "encrypt", 0, 0, 'E' },
|
|
{ "secure", 0, 0, 'S' },
|
|
{ "master", 0, 0, 'M' },
|
|
{ "linger", 1, 0, 'L' },
|
|
{ 0, 0, 0, 0 }
|
|
};
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
bdaddr_t bdaddr;
|
|
int i, opt, ctl, dev_id, show_all = 0;
|
|
|
|
bacpy(&bdaddr, BDADDR_ANY);
|
|
|
|
while ((opt = getopt_long(argc, argv, "+i:f:rahAESML:", main_options, NULL)) != -1) {
|
|
switch(opt) {
|
|
case 'i':
|
|
if (strncmp(optarg, "hci", 3) == 0)
|
|
hci_devba(atoi(optarg + 3), &bdaddr);
|
|
else
|
|
str2ba(optarg, &bdaddr);
|
|
break;
|
|
|
|
case 'f':
|
|
rfcomm_config_file = strdup(optarg);
|
|
break;
|
|
|
|
case 'r':
|
|
rfcomm_raw_tty = 1;
|
|
break;
|
|
|
|
case 'a':
|
|
show_all = 1;
|
|
break;
|
|
|
|
case 'h':
|
|
usage();
|
|
exit(0);
|
|
|
|
case 'A':
|
|
auth = 1;
|
|
break;
|
|
|
|
case 'E':
|
|
encryption = 1;
|
|
break;
|
|
|
|
case 'S':
|
|
secure = 1;
|
|
break;
|
|
|
|
case 'M':
|
|
master = 1;
|
|
break;
|
|
|
|
case 'L':
|
|
linger = atoi(optarg);
|
|
break;
|
|
|
|
default:
|
|
exit(0);
|
|
}
|
|
}
|
|
|
|
argc -= optind;
|
|
argv += optind;
|
|
optind = 0;
|
|
|
|
if (argc < 2) {
|
|
if (argc != 0) {
|
|
usage();
|
|
exit(1);
|
|
} else
|
|
show_all = 1;
|
|
}
|
|
|
|
ctl = socket(AF_BLUETOOTH, SOCK_RAW, BTPROTO_RFCOMM);
|
|
if (ctl < 0) {
|
|
perror("Can't open RFCOMM control socket");
|
|
exit(1);
|
|
}
|
|
|
|
if (show_all) {
|
|
print_dev_list(ctl, 0);
|
|
close(ctl);
|
|
exit(0);
|
|
}
|
|
|
|
if (strncmp(argv[1], "/dev/rfcomm", 11) == 0)
|
|
dev_id = atoi(argv[1] + 11);
|
|
else if (strncmp(argv[1], "rfcomm", 6) == 0)
|
|
dev_id = atoi(argv[1] + 6);
|
|
else
|
|
dev_id = atoi(argv[1]);
|
|
|
|
for (i = 0; command[i].cmd; i++) {
|
|
if (strncmp(command[i].cmd, argv[0], 4) && strncmp(command[i].alt, argv[0], 4))
|
|
continue;
|
|
argc--;
|
|
argv++;
|
|
command[i].func(ctl, dev_id, &bdaddr, argc, argv);
|
|
close(ctl);
|
|
exit(0);
|
|
}
|
|
|
|
usage();
|
|
|
|
close(ctl);
|
|
|
|
return 0;
|
|
}
|