/****************************************************************************** * * Copyright (C) 2013 Google, Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ******************************************************************************/ #define LOG_TAG "btsnoop_net" #include #include #include #include #include #include #include #include #include #include #include "osi.h" static void safe_close_(int *fd); static void *listen_fn_(void *context); static const char *LISTEN_THREAD_NAME_ = "btsnoop_net_listen"; static const int LOCALHOST_ = 0x7F000001; static const int LISTEN_PORT_ = 8872; static pthread_t listen_thread_; static bool listen_thread_valid_ = false; static pthread_mutex_t client_socket_lock_ = PTHREAD_MUTEX_INITIALIZER; static int listen_socket_ = -1; static int client_socket_ = -1; void btsnoop_net_open() { listen_thread_valid_ = (pthread_create(&listen_thread_, NULL, listen_fn_, NULL) == 0); if (!listen_thread_valid_) { ALOGE("%s pthread_create failed: %s", __func__, strerror(errno)); } else { ALOGD("initialized"); } } void btsnoop_net_close() { if (listen_thread_valid_) { shutdown(listen_socket_, SHUT_RDWR); pthread_join(listen_thread_, NULL); safe_close_(&client_socket_); listen_thread_valid_ = false; } } void btsnoop_net_write(const void *data, size_t length) { pthread_mutex_lock(&client_socket_lock_); if (client_socket_ != -1) { if (send(client_socket_, data, length, 0) == -1 && errno == ECONNRESET) { safe_close_(&client_socket_); } } pthread_mutex_unlock(&client_socket_lock_); } static void *listen_fn_(UNUSED_ATTR void *context) { prctl(PR_SET_NAME, (unsigned long)LISTEN_THREAD_NAME_, 0, 0, 0); listen_socket_ = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (listen_socket_ == -1) { ALOGE("%s socket creation failed: %s", __func__, strerror(errno)); goto cleanup; } int enable = 1; if (setsockopt(listen_socket_, SOL_SOCKET, SO_REUSEADDR, &enable, sizeof(enable)) == -1) { ALOGE("%s unable to set SO_REUSEADDR: %s", __func__, strerror(errno)); goto cleanup; } struct sockaddr_in addr; addr.sin_family = AF_INET; addr.sin_addr.s_addr = htonl(LOCALHOST_); addr.sin_port = htons(LISTEN_PORT_); if (bind(listen_socket_, (struct sockaddr *)&addr, sizeof(addr)) == -1) { ALOGE("%s unable to bind listen socket: %s", __func__, strerror(errno)); goto cleanup; } if (listen(listen_socket_, 10) == -1) { ALOGE("%s unable to listen: %s", __func__, strerror(errno)); goto cleanup; } for (;;) { ALOGD("waiting for client connection"); int client_socket = accept(listen_socket_, NULL, NULL); if (client_socket == -1) { if (errno == EINVAL || errno == EBADF) { break; } ALOGW("%s error accepting socket: %s", __func__, strerror(errno)); continue; } /* When a new client connects, we have to send the btsnoop file header. This allows a decoder to treat the session as a new, valid btsnoop file. */ ALOGI("client connected"); pthread_mutex_lock(&client_socket_lock_); safe_close_(&client_socket_); client_socket_ = client_socket; send(client_socket_, "btsnoop\0\0\0\0\1\0\0\x3\xea", 16, 0); pthread_mutex_unlock(&client_socket_lock_); } cleanup: safe_close_(&listen_socket_); return NULL; } static void safe_close_(int *fd) { assert(fd != NULL); if (*fd != -1) { close(*fd); *fd = -1; } }