| /* |
| * |
| * BlueZ - Bluetooth protocol stack for Linux |
| * |
| * Copyright (C) 2017 Intel Corporation. All rights reserved. |
| * |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library 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 |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; 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 |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <stdbool.h> |
| #include <inttypes.h> |
| |
| #include <glib.h> |
| |
| #include <lib/bluetooth.h> |
| |
| #include "src/shared/shell.h" |
| #include "mesh/util.h" |
| #include "mesh/agent.h" |
| |
| struct input_request { |
| oob_type_t type; |
| uint16_t len; |
| agent_input_cb cb; |
| void *user_data; |
| }; |
| |
| static struct input_request pending_request = {NONE, 0, NULL, NULL}; |
| |
| bool agent_completion(void) |
| { |
| if (pending_request.type == NONE) |
| return false; |
| |
| return true; |
| } |
| |
| static void reset_input_request(void) |
| { |
| pending_request.type = NONE; |
| pending_request.len = 0; |
| pending_request.cb = NULL; |
| pending_request.user_data = NULL; |
| } |
| |
| static void response_hexadecimal(const char *input, void *user_data) |
| { |
| uint8_t buf[MAX_HEXADECIMAL_OOB_LEN]; |
| |
| if (!str2hex(input, strlen(input), buf, pending_request.len) ) { |
| bt_shell_printf("Incorrect input: expecting %d hex octets\n", |
| pending_request.len); |
| return; |
| } |
| |
| if (pending_request.cb) |
| pending_request.cb(HEXADECIMAL, buf, pending_request.len, |
| pending_request.user_data); |
| |
| reset_input_request(); |
| } |
| |
| static void response_decimal(const char *input, void *user_data) |
| { |
| uint8_t buf[DECIMAL_OOB_LEN]; |
| |
| if (strlen(input) > pending_request.len) |
| return; |
| |
| bt_put_be32(atoi(input), buf); |
| |
| if (pending_request.cb) |
| pending_request.cb(DECIMAL, buf, DECIMAL_OOB_LEN, |
| pending_request.user_data); |
| |
| reset_input_request(); |
| } |
| |
| static void response_ascii(const char *input, void *user_data) |
| { |
| if (pending_request.cb) |
| pending_request.cb(ASCII, (uint8_t *) input, strlen(input), |
| pending_request.user_data); |
| |
| reset_input_request(); |
| } |
| |
| static bool request_hexadecimal(uint16_t len) |
| { |
| if (len > MAX_HEXADECIMAL_OOB_LEN) |
| return false; |
| |
| bt_shell_printf("Request hexadecimal key (hex %d octets)\n", len); |
| bt_shell_prompt_input("mesh", "Enter key (hex number):", response_hexadecimal, |
| NULL); |
| |
| return true; |
| } |
| |
| static uint32_t power_ten(uint8_t power) |
| { |
| uint32_t ret = 1; |
| |
| while (power--) |
| ret *= 10; |
| |
| return ret; |
| } |
| |
| static bool request_decimal(uint16_t len) |
| { |
| bt_shell_printf("Request decimal key (0 - %d)\n", power_ten(len) - 1); |
| bt_shell_prompt_input("mesh", "Enter Numeric key:", response_decimal, NULL); |
| |
| return true; |
| } |
| |
| static bool request_ascii(uint16_t len) |
| { |
| if (len > MAX_ASCII_OOB_LEN) |
| return false; |
| |
| bt_shell_printf("Request ASCII key (max characters %d)\n", len); |
| bt_shell_prompt_input("mesh", "Enter key (ascii string):", response_ascii, |
| NULL); |
| |
| return true; |
| } |
| |
| bool agent_input_request(oob_type_t type, uint16_t max_len, agent_input_cb cb, |
| void *user_data) |
| { |
| bool result; |
| |
| if (pending_request.type != NONE) |
| return FALSE; |
| |
| switch (type) { |
| case HEXADECIMAL: |
| result = request_hexadecimal(max_len); |
| break; |
| case DECIMAL: |
| result = request_decimal(max_len); |
| break; |
| case ASCII: |
| result = request_ascii(max_len); |
| break; |
| case NONE: |
| case OUTPUT: |
| default: |
| return false; |
| }; |
| |
| if (result) { |
| pending_request.type = type; |
| pending_request.len = max_len; |
| pending_request.cb = cb; |
| pending_request.user_data = user_data; |
| |
| return true; |
| } |
| |
| return false; |
| } |
| |
| static void response_output(const char *input, void *user_data) |
| { |
| reset_input_request(); |
| } |
| |
| bool agent_output_request(const char* str) |
| { |
| if (pending_request.type != NONE) |
| return false; |
| |
| pending_request.type = OUTPUT; |
| bt_shell_prompt_input("mesh", str, response_output, NULL); |
| return true; |
| } |
| |
| void agent_output_request_cancel(void) |
| { |
| if (pending_request.type != OUTPUT) |
| return; |
| pending_request.type = NONE; |
| bt_shell_release_prompt(""); |
| } |