• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //
2 // Copyright 2015 The Android Open Source Project
3 //
4 // Licensed under the Apache License, Version 2.0 (the "License");
5 // you may not use this file except in compliance with the License.
6 // You may obtain a copy of the License at
7 //
8 // http://www.apache.org/licenses/LICENSE-2.0
9 //
10 // Unless required by applicable law or agreed to in writing, software
11 // distributed under the License is distributed on an "AS IS" BASIS,
12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 // See the License for the specific language governing permissions and
14 // limitations under the License.
15 //
16 
17 #define LOG_TAG "test_channel_transport"
18 
19 #include "vendor_libs/test_vendor_lib/include/test_channel_transport.h"
20 
21 #include "base/logging.h"
22 
23 extern "C" {
24 #include "osi/include/log.h"
25 
26 #include <sys/socket.h>
27 #include <netinet/in.h>
28 }  // extern "C"
29 
30 namespace test_vendor_lib {
31 
TestChannelTransport(bool enabled,int port)32 TestChannelTransport::TestChannelTransport(bool enabled, int port)
33     : enabled_(enabled), port_(port) {}
34 
SetUp()35 bool TestChannelTransport::SetUp() {
36   CHECK(enabled_);
37 
38   struct sockaddr_in listen_address, test_channel_address;
39   int sockaddr_in_size = sizeof(struct sockaddr_in);
40   int listen_fd = -1;
41   int accept_fd = -1;
42   memset(&listen_address, 0, sockaddr_in_size);
43   memset(&test_channel_address, 0, sockaddr_in_size);
44 
45   if ((listen_fd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
46     LOG_INFO(LOG_TAG, "Error creating socket for test channel.");
47     return false;
48   }
49 
50   LOG_INFO(LOG_TAG, "port: %d", port_);
51   listen_address.sin_family = AF_INET;
52   listen_address.sin_port = htons(port_);
53   listen_address.sin_addr.s_addr = htonl(INADDR_ANY);
54 
55   if (bind(listen_fd, reinterpret_cast<sockaddr*>(&listen_address),
56            sockaddr_in_size) < 0) {
57     LOG_INFO(LOG_TAG, "Error binding test channel listener socket to address.");
58     close(listen_fd);
59     return false;
60   }
61 
62   if (listen(listen_fd, 1) < 0) {
63     LOG_INFO(LOG_TAG, "Error listening for test channel.");
64     close(listen_fd);
65     return false;
66   }
67 
68   if ((accept_fd =
69            accept(listen_fd, reinterpret_cast<sockaddr*>(&test_channel_address),
70                   &sockaddr_in_size)) < 0) {
71     LOG_INFO(LOG_TAG, "Error accepting test channel connection.");
72     close(listen_fd);
73     return false;
74   }
75 
76   fd_.reset(new base::ScopedFD(accept_fd));
77   return GetFd() >= 0;
78 }
79 
GetFd()80 int TestChannelTransport::GetFd() {
81   return fd_->get();
82 }
83 
IsEnabled()84 bool TestChannelTransport::IsEnabled() {
85   return enabled_;
86 }
87 
88 // base::MessageLoopForIO::Watcher overrides:
OnFileCanReadWithoutBlocking(int fd)89 void TestChannelTransport::OnFileCanReadWithoutBlocking(int fd) {
90   CHECK(fd == GetFd());
91 
92   LOG_INFO(LOG_TAG, "Event ready in TestChannelTransport on fd: %d", fd);
93   uint8_t command_name_size = 0;
94   read(fd, &command_name_size, 1);
95   std::vector<uint8_t> command_name_raw;
96   command_name_raw.resize(command_name_size);
97   read(fd, &command_name_raw[0], command_name_size);
98   std::string command_name(command_name_raw.begin(), command_name_raw.end());
99   LOG_INFO(LOG_TAG, "Received command from test channel: %s",
100            command_name.data());
101 
102   if (command_name == "CLOSE_TEST_CHANNEL") {
103     fd_.reset(nullptr);
104     return;
105   }
106 
107   uint8_t num_args = 0;
108   read(fd, &num_args, 1);
109   LOG_INFO(LOG_TAG, "num_args: %d", num_args);
110   std::vector<std::string> args;
111   for (uint8_t i = 0; i < num_args; ++i) {
112     uint8_t arg_size = 0;
113     read(fd, &arg_size, 1);
114     std::vector<uint8_t> arg;
115     arg.resize(arg_size);
116     read(fd, &arg[0], arg_size);
117     args.push_back(std::string(arg.begin(), arg.end()));
118   }
119 
120   for (size_t i = 0; i < args.size(); ++i)
121     LOG_INFO(LOG_TAG, "Command argument %zu: %s", i, args[i].data());
122 
123   command_handler_(command_name, args);
124 }
125 
OnFileCanWriteWithoutBlocking(int fd)126 void TestChannelTransport::OnFileCanWriteWithoutBlocking(int fd) {}
127 
RegisterCommandHandler(std::function<void (const std::string &,const std::vector<std::string> &)> callback)128 void TestChannelTransport::RegisterCommandHandler(
129     std::function<void(const std::string&, const std::vector<std::string>&)>
130         callback) {
131   command_handler_ = callback;
132 }
133 
Disable()134 void TestChannelTransport::Disable() {
135   enabled_ = false;
136 }
137 
138 }  // namespace test_vendor_lib {
139