📄 main.c
字号:
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); 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;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -