4 * Copyright (c) 2015 Samsung Electronics Co., Ltd.
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at
10 * http://www.apache.org/licenses/LICENSE-2.0
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
20 #include <sensor_common.h>
21 #include <command_common.h>
22 #include <external_sensor_worker.h>
23 #include <external_client_manager.h>
24 #include <external_sensor.h>
25 #include <external_sensor_service.h>
29 external_sensor_worker::cmd_handler_t external_sensor_worker::m_cmd_handlers[];
31 external_sensor_worker::external_sensor_worker(const csocket& socket)
32 : m_client_id(CLIENT_ID_INVALID)
35 , m_sensor_id(UNKNOWN_SENSOR)
37 static bool init = false;
44 m_worker.set_context(this);
45 m_worker.set_working(working);
46 m_worker.set_stopped(stopped);
49 external_sensor_worker::~external_sensor_worker()
54 bool external_sensor_worker::start(void)
56 return m_worker.start();
59 void external_sensor_worker::init_cmd_handlers(void)
61 m_cmd_handlers[CMD_EXT_GET_ID] = &external_sensor_worker::cmd_get_id;
62 m_cmd_handlers[CMD_EXT_CONNECT] = &external_sensor_worker::cmd_connect;
63 m_cmd_handlers[CMD_EXT_DISCONNECT] = &external_sensor_worker::cmd_disconnect;
64 m_cmd_handlers[CMD_EXT_POST] = &external_sensor_worker::cmd_post;
67 bool external_sensor_worker::working(void *ctx)
70 external_sensor_worker *inst = (external_sensor_worker *)ctx;
75 if (inst->m_socket.recv(&header, sizeof(header)) <= 0) {
78 _D("%s failed to receive header", info.c_str());
82 retvm_if(header.size == 0, false, "Invalid header size");
84 payload = new(std::nothrow) char[header.size];
85 retvm_if(!payload, false, "Failed to allocate memory");
87 if (inst->m_socket.recv(payload, header.size) <= 0) {
90 _D("%s failed to receive data of packet", info.c_str());
95 ret = inst->dispatch_command(header.cmd, payload);
101 bool external_sensor_worker::stopped(void *ctx)
104 external_sensor_worker *inst = (external_sensor_worker *)ctx;
106 inst->get_info(info);
107 _I("%s is stopped", info.c_str());
109 if (inst->m_client_id != CLIENT_ID_INVALID) {
110 _I("Client:%d leaves without disconnecting", inst->m_client_id);
111 if (get_client_manager().has_sensor_record(inst->m_client_id, inst->m_sensor_id)) {
112 _I("Removing sensor[%#x] record for client_id[%d]", inst->m_sensor_id, inst->m_client_id);
113 get_client_manager().remove_sensor_record(inst->m_client_id, inst->m_sensor_id);
116 inst->m_sensor->set_source_connected(false);
124 bool external_sensor_worker::dispatch_command(int cmd, void* payload)
128 if (!(cmd > 0 && cmd < CMD_EXT_CNT)) {
129 _E("Unknown command: %d", cmd);
131 cmd_handler_t cmd_handler;
132 cmd_handler = external_sensor_worker::m_cmd_handlers[cmd];
134 ret = (this->*cmd_handler)(payload);
140 bool external_sensor_worker::send_cmd_done(long value)
143 cmd_ext_done_t *cmd_ext_done;
145 ret_packet = new(std::nothrow) cpacket(sizeof(cmd_ext_done_t));
146 retvm_if(!ret_packet, false, "Failed to allocate memory");
148 ret_packet->set_cmd(CMD_EXT_DONE);
150 cmd_ext_done = (cmd_ext_done_t*)ret_packet->data();
151 cmd_ext_done->value = value;
153 if (m_socket.send(ret_packet->packet(), ret_packet->size()) <= 0) {
154 _E("Failed to send a cmd_done to client_id [%d] with value [%ld]", m_client_id, value);
163 bool external_sensor_worker::send_cmd_get_id_done(int client_id)
166 cmd_ext_get_id_done_t *cmd_ext_get_id_done;
168 ret_packet = new(std::nothrow) cpacket(sizeof(cmd_ext_get_id_done_t));
169 retvm_if(!ret_packet, false, "Failed to allocate memory");
171 ret_packet->set_cmd(CMD_EXT_GET_ID);
173 cmd_ext_get_id_done = (cmd_ext_get_id_done_t*)ret_packet->data();
174 cmd_ext_get_id_done->client_id = client_id;
176 if (m_socket.send(ret_packet->packet(), ret_packet->size()) <= 0) {
177 _E("Failed to send a cmd_get_id_done with client_id [%d]", client_id);
186 bool external_sensor_worker::send_cmd_connect_done(sensor_id_t sensor_id)
189 cmd_ext_connect_done_t *cmd_ext_connect_done;
191 ret_packet = new(std::nothrow) cpacket(sizeof(cmd_ext_connect_done_t));
192 retvm_if(!ret_packet, false, "Failed to allocate memory");
194 ret_packet->set_cmd(CMD_EXT_CONNECT);
196 cmd_ext_connect_done = (cmd_ext_connect_done_t*)ret_packet->data();
197 cmd_ext_connect_done->sensor_id = sensor_id;
199 if (m_socket.send(ret_packet->packet(), ret_packet->size()) <= 0) {
200 _E("Failed to send a cmd_connect done");
209 bool external_sensor_worker::cmd_get_id(void *payload)
211 cmd_ext_get_id_t *cmd = static_cast<cmd_ext_get_id_t *>(payload);
214 socklen_t opt_len = sizeof(cr);
216 if (getsockopt(m_socket.get_socket_fd(), SOL_SOCKET, SO_PEERCRED, &cr, &opt_len)) {
217 _E("Failed to get socket option with SO_PEERCRED");
221 client_id = get_client_manager().create_client_record();
223 if (client_id != MAX_HANDLE_REACHED) {
224 get_client_manager().set_client_info(client_id, cr.pid, cmd->name);
225 _I("New client id [%d] created", client_id);
228 if (!send_cmd_get_id_done(client_id))
229 _E("Failed to send cmd_done to a client");
234 bool external_sensor_worker::cmd_connect(void *payload)
236 cmd_ext_connect_t *cmd = static_cast<cmd_ext_connect_t *>(payload);
237 m_client_id = cmd->client_id;
239 external_sensor *sensor;
240 sensor = external_sensor_service::get_instance().get_sensor(string(cmd->key));
242 _E("No matched external sensor with key: %s", cmd->key);
246 if (!sensor->set_source_connected(true)) {
247 _E("External sensor(%s) is already connected", cmd->key);
252 m_sensor_id = sensor->get_id();
254 if (!get_client_manager().create_sensor_record(m_client_id, m_sensor_id)) {
255 _E("Failed to create sensor record for client: %d, sensor_id: %d", m_client_id, m_sensor_id);
256 m_sensor_id = UNKNOWN_SENSOR;
261 if (!send_cmd_connect_done(m_sensor_id))
262 _E("Failed to send cmd_connect_done to a client : %d", m_client_id);
267 bool external_sensor_worker::cmd_disconnect(void *payload)
269 long ret_value = OP_ERROR;
272 _E("External sensor is not connected");
273 ret_value = OP_ERROR;
277 if (!get_client_manager().remove_sensor_record(m_client_id, m_sensor_id)) {
278 _E("Failed to remove sensor record for client [%d]", m_client_id);
279 ret_value = OP_ERROR;
283 m_sensor->set_source_connected(false);
286 m_client_id = CLIENT_ID_INVALID;
287 m_sensor_id = UNKNOWN_SENSOR;
288 ret_value = OP_SUCCESS;
290 if (!send_cmd_done(ret_value))
291 _E("Failed to send cmd_done to a client");
293 if (ret_value == OP_SUCCESS)
299 bool external_sensor_worker::cmd_post(void *payload)
301 long ret_value = OP_SUCCESS;
302 cmd_ext_post_t *cmd = static_cast<cmd_ext_post_t *>(payload);
304 m_sensor->on_receive(cmd->timestamp, cmd->data, cmd->data_cnt);
306 if (!send_cmd_done(ret_value))
307 _E("Failed to send cmd_done to a client");
312 external_client_manager& external_sensor_worker::get_client_manager(void)
314 return external_client_manager::get_instance();
317 void external_sensor_worker::get_info(string &info)
319 const char *client_info = NULL;
321 if (m_client_id != CLIENT_ID_INVALID)
322 client_info = get_client_manager().get_client_info(m_client_id);
324 info = string("Command worker for ") + (client_info ? string(client_info) : string("Unknown"));