#include <stdint.h>
// Copyright 2015-2016 Espressif Systems (Shanghai) PTE LTD
//
// 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.

#include <string.h>
#include <stdbool.h>
#include <stdio.h>

//#include "bluedroid_test.h"
#include "bta/bta_api.h"
#include "bta/bta_gatt_api.h"
#include "device/controller.h"

#include "gatt_int.h"
#include "common/bt_trace.h"
#include "stack/btm_api.h"
#include "stack/bt_types.h"
#include "dis_api.h"

#if BLE_INCLUDED == true

#define BA_MAX_CHAR_NUM     1
#define BA_MAX_ATTR_NUM     (BA_MAX_CHAR_NUM * 5 + 1)
/*max 3 descriptors, 1 desclaration and 1 value*/

#ifndef BATTER_LEVEL_PROP
#define BATTER_LEVEL_PROP   (GATT_CHAR_PROP_BIT_READ|GATT_CHAR_PROP_BIT_NOTIFY)
#endif

#ifndef BATTER_LEVEL_PERM
#define BATTER_LEVEL_PERM   (GATT_PERM_READ)
#endif

#define BT_BD_ADDR_STR         "%02x:%02x:%02x:%02x:%02x:%02x"
#define BT_BD_ADDR_HEX(addr)   addr[0], addr[1], addr[2], addr[3], addr[4], addr[5]
esp_gatts_if_t server_if;

tBATTERY_CB         battery_cb;
tGATT_CHAR_PROP     prop = GATT_CHAR_PROP_BIT_READ;
tBA_REG_INFO        ba_reg_info;
UINT8               attr_handle_bit = 0x00;

extern tDIS_CB      dis_cb;
esp_bt_uuid_t       bas_uuid = {LEN_UUID_16, {UUID_SERVCLASS_BATTERY}};
/******************************************************************************
** Function      bas_gatts_callback
**
** Description   battery service register callback function
*******************************************************************************/
static void bas_gatts_callback(esp_gatts_evt_t event, tBTA_GATTS *p_data)
{
    switch (event) {
    case ESP_GATTS_REG_EVT: {
        esp_gatt_status_t  status = p_data->reg_oper.status;
        server_if = p_data->reg_oper.server_if;
        BTC_TRACE_ERROR("BAS register completed: event=%d, status=%d, server_if=%d\n",
                  event, status, server_if);

        UINT8 app_id = 0xff;
        bas_init(server_if, app_id);

        tDIS_ATTR_MASK mask = 0x01ff;
        DIS_Init(server_if, mask);
    }
    break;

    /*connect callback*/
    case ESP_GATTS_CONNECT_EVT: {
        BTC_TRACE_ERROR("\ndevice is connected "BT_BD_ADDR_STR", server_if=%d,reason=0x%x,connect_id=%d\n",
                  BT_BD_ADDR_HEX(p_data->conn.remote_bda), p_data->conn.server_if,
                  p_data->conn.reason, p_data->conn.conn_id);
        /*return whether the remote device is currently connected*/
        int is_connected = BTA_DmGetConnectionState(p_data->conn.remote_bda);
        BTC_TRACE_ERROR("is_connected=%d\n", is_connected);
    }
    break;

    /*create service callback*/
    case ESP_GATTS_CREATE_EVT: {
        BTC_TRACE_ERROR("create service:server_if=%d,service_id=0x%x,service_uuid=0x%x\n",
                  p_data->create.server_if, p_data->create.service_id,
                  p_data->create.uuid.uu.uuid16);
        UINT16 service_uuid = p_data->create.uuid.uu.uuid16;
        UINT16 service_id = p_data->create.service_id;
        if (service_uuid == 0x180f) {
            tBT_UUID   uuid = {LEN_UUID_16, {GATT_UUID_BATTERY_LEVEL}};
            bas_AddChar(service_id, &uuid);
        }
        if (service_uuid == 0x180a) {
            dis_cb.service_handle = service_id;
            dis_cb.max_handle = service_id + DIS_MAX_ATTR_NUM;
            dis_AddChar(service_id);
        }

    }
    break;

    case ESP_GATTS_ADD_CHAR_EVT: {
        BTC_TRACE_ERROR("create characteristic:server_if=%d,service_id=0x%x,char_uuid=0x%x\n",
                  p_data->add_result.server_if, p_data->add_result.service_id,
                  p_data->add_result.char_uuid.uu.uuid16);
        UINT16 char_uuid = p_data->add_result.char_uuid.uu.uuid16;
        UINT16 service_id = p_data->add_result.service_id;
        if (char_uuid == GATT_UUID_BATTERY_LEVEL) {
            bas_AddCharDescr(service_id, p_data->add_result.attr_id);
        }
        if (char_uuid == GATT_UUID_SYSTEM_ID | GATT_UUID_MODEL_NUMBER_STR | GATT_UUID_PNP_ID |
                GATT_UUID_SERIAL_NUMBER_STR | GATT_UUID_FW_VERSION_STR | GATT_UUID_HW_VERSION_STR |
                GATT_UUID_SW_VERSION_STR | GATT_UUID_MANU_NAME | GATT_UUID_IEEE_DATA) {
            switch (char_uuid) {
            case GATT_UUID_SYSTEM_ID:
                dis_cb.dis_attr[0].handle = service_id; break;
            case GATT_UUID_MODEL_NUMBER_STR:
                dis_cb.dis_attr[1].handle = service_id; break;
            case GATT_UUID_SERIAL_NUMBER_STR:
                dis_cb.dis_attr[2].handle = service_id; break;
            case GATT_UUID_FW_VERSION_STR:
                dis_cb.dis_attr[3].handle = service_id; break;
            case GATT_UUID_HW_VERSION_STR:
                dis_cb.dis_attr[4].handle = service_id; break;
            case GATT_UUID_SW_VERSION_STR:
                dis_cb.dis_attr[5].handle = service_id; break;
            case GATT_UUID_MANU_NAME:
                dis_cb.dis_attr[6].handle = service_id; break;
            case GATT_UUID_IEEE_DATA:
                dis_cb.dis_attr[7].handle = service_id; break;
            case GATT_UUID_PNP_ID:
                dis_cb.dis_attr[8].handle = service_id; break;
            }
        }
    }
    break;

    case ESP_GATTS_ADD_CHAR_DESCR_EVT: {

        BTC_TRACE_ERROR("create descriptor:server_if=%d,service_id=0x%x,attr_id=0x%x,char_uuid=0x%x\n",
                  p_data->add_result.server_if, p_data->add_result.service_id,
                  p_data->add_result.attr_id, p_data->add_result.char_uuid.uu.uuid16);
        bas_AddCharDescr(p_data->add_result.service_id, p_data->add_result.attr_id);
    }
    break;

    case ESP_GATTS_START_EVT: {
        BTC_TRACE_ERROR("start service:server_if=%d,service_id=0x%x\n", p_data->srvc_oper.server_if,
                  p_data->srvc_oper.service_id);
        bas_service_cmpl(p_data->srvc_oper.service_id, p_data->srvc_oper.status);

        /*start advertising*/
        //if(p_data->srvc_oper.status == GATT_SUCCESS)
        // BTA_GATTS_Listen(server_if, true, NULL);
        //    BTA_GATTC_Broadcast(client_if, true);       //non-connectable
    }
    break;

    case ESP_GATTS_READ_EVT: {
        UINT32 trans_id = p_data->req_data.trans_id;
        UINT16 conn_id = p_data->req_data.conn_id;
        UINT16 handle = p_data->req_data.p_data->read_req.handle;
        bool   is_long = p_data->req_data.p_data->read_req.is_long;
        BTC_TRACE_ERROR("read request:event=0x%x,handle=0x%x,trans_id=0x%x,conn_id=0x%x\n",
                  event, handle, trans_id, conn_id);

        if (dis_valid_handle_range(handle)) {
            tGATT_VALUE p_value;
            p_value.handle = handle;
            p_value.conn_id = conn_id;
            p_value.offset = p_data->req_data.p_data->read_req.offset;
            dis_s_read_attr_value(p_data->req_data.p_data, &p_value, trans_id, conn_id);
        } else {
            bas_s_read_attr_value(p_data->req_data.p_data, trans_id, conn_id);
        }
    }
    break;

    case ESP_GATTS_WRITE_EVT: {

        UINT32 trans_id = p_data->req_data.trans_id;
        UINT16 conn_id = p_data->req_data.conn_id;
        UINT16 handle = p_data->req_data.p_data->write_req.handle;
        BTC_TRACE_ERROR("write request:event=0x%x,handle=0x%x,trans_id=0x%x,conn_id=0x%x\n",
                  event, handle, trans_id, conn_id);
        bas_s_write_attr_value(p_data->req_data.p_data, trans_id, conn_id,
                               p_data->req_data.remote_bda);
    }
    break;

    case ESP_GATTS_EXEC_WRITE_EVT: {
        UINT32 trans_id = p_data->req_data.trans_id;
        UINT16 conn_id = p_data->req_data.conn_id;
        UINT8  exec_write = p_data->req_data.p_data->exec_write;
        BTC_TRACE_ERROR("execute write request:event=0x%x,exce_write=0x%x,trans_id=0x%x,conn_id=0x%x\n",
                  event, exec_write, trans_id, conn_id);
    }
    break;

    case ESP_GATTS_MTU_EVT: {
        UINT32 trans_id = p_data->req_data.trans_id;
        UINT16 conn_id = p_data->req_data.conn_id;
        UINT16 mtu = p_data->req_data.p_data->mtu;
        BTC_TRACE_ERROR("exchange mtu request:event=0x%x,mtu=0x%x,trans_id=0x%x,conn_id=0x%x\n",
                  event, mtu, trans_id, conn_id);
    }
    break;

    case ESP_GATTS_CFM_EVT: {

        UINT32 trans_id = p_data->req_data.trans_id;
        UINT16 conn_id = p_data->req_data.conn_id;
        BTC_TRACE_ERROR("configue request:trans_id=0x%x,conn_id=0x%x\n",
                  trans_id, conn_id);
    }
    break;

    default:
        BTC_TRACE_ERROR("unsettled event: %d\n", event);
        break;
    }

}
/******************************************************************************
** Function      bas_callback
**
** Description   battery service callback for client request
*******************************************************************************/
static void bas_callback(UINT32 trans_id, UINT16 conn_id, UINT8 app_id,
                         UINT8 event, tBA_WRITE_DATA *p_data)
{
    tBA_RSP_DATA p_rsp;
    tGATT_STATUS st = ESP_GATT_OK;
    switch (event) {
    case BA_READ_LEVEL_REQ : {
        BTC_TRACE_ERROR("read battery level\n");
        p_rsp.ba_level = 60;     //battery level
        Battery_Rsp(trans_id, conn_id, app_id, st, event, &p_rsp);
    }
    break;

    case BA_READ_PRE_FMT_REQ : {
        BTC_TRACE_ERROR("read presentation format\n");
    }
    break;

    case BA_READ_CLT_CFG_REQ : {
        BTC_TRACE_ERROR("read client characteristic configuration request\n");
        p_rsp.clt_cfg = 0x0001;   //notification
        Battery_Rsp(trans_id, conn_id, app_id, st, event, &p_rsp);
    }
    break;

    case BA_READ_RPT_REF_REQ : {
        BTC_TRACE_ERROR("read report reference descriptor\n");
    }
    break;

    /*battery level notify*/
    case BA_WRITE_CLT_CFG_REQ : {
        BTC_TRACE_ERROR("write client characteristic configuration request\n");
        Battery_Rsp(trans_id, conn_id, app_id, st, event, NULL);

        int battery_level = 50;
        Battery_Notify(conn_id, app_id, p_data->remote_bda, battery_level);
    }
    break;

    default:
        break;
    }

    return;
}
/*****************************************************************************
** Function      bas_s_read_attr_value
**
** Description   it will be called when client sends a read request
******************************************************************************/
void bas_s_read_attr_value(tGATTS_DATA *p_data, UINT32 trans_id, UINT16 conn_id)
{

    tBA_INST            *p_inst = &battery_cb.battery_inst[0];
    UINT8               i;
    esp_gatt_status_t   st = ESP_GATT_NOT_FOUND;
    UINT16              handle = p_data->read_req.handle;


    for (i = 0; i < BA_MAX_INT_NUM; i ++, p_inst ++) {
        // read battery level
        if (handle == p_inst->ba_level_hdl ||
                handle == p_inst->clt_cfg_hdl ||
                handle == p_inst->rpt_ref_hdl ||
                handle == p_inst->pres_fmt_hdl) {
            if (p_data->read_req.is_long) {
                st = ESP_GATT_NOT_LONG;
            }

            if (p_inst->p_cback) {
                if (handle == p_inst->ba_level_hdl) {
                    p_inst->pending_evt = BA_READ_LEVEL_REQ;
                }
                if (handle == p_inst->clt_cfg_hdl) {
                    p_inst->pending_evt = BA_READ_CLT_CFG_REQ;
                }
                if (handle == p_inst->pres_fmt_hdl) {
                    p_inst->pending_evt = BA_READ_PRE_FMT_REQ;
                }
                if (handle == p_inst->rpt_ref_hdl) {
                    p_inst->pending_evt = BA_READ_RPT_REF_REQ ;
                }

                // p_inst->pending_clcb_idx = clcb_idx;
                p_inst->pending_handle = handle;
                //act = SRVC_ACT_PENDING;
                (*p_inst->p_cback)(trans_id, conn_id, p_inst->app_id, p_inst->pending_evt, NULL);
            } else { /* application is not registered */
                st = ESP_GATT_ERR_UNLIKELY;
            }
            break;
        }
        /* else attribute not found */
    }
}

/*****************************************************************************
** Function      bas_s_write_attr_value
**
** Description   it will be called when client sends a write request
******************************************************************************/
void bas_s_write_attr_value(tGATTS_DATA *p_data, UINT32 trans_id, UINT16 conn_id, BD_ADDR bd_addr)
{
    tBA_WRITE_DATA   cfg;
    UINT8            *p = p_data->write_req.value;
    tBA_INST         *p_inst = &battery_cb.battery_inst[0];
    UINT8             i;
    esp_gatt_status_t st = ESP_GATT_NOT_FOUND;
    UINT16            handle = p_data->write_req.handle;


    for (i = 0; i < BA_MAX_INT_NUM; i ++, p_inst ++) {
        if (handle == p_inst->clt_cfg_hdl) {
            memcpy(cfg.remote_bda, bd_addr, BD_ADDR_LEN);
            STREAM_TO_UINT16(cfg.clt_cfg, p);

            if (p_inst->p_cback) {
                p_inst->pending_evt = BA_WRITE_CLT_CFG_REQ;
                p_inst->pending_handle = handle;
                cfg.need_rsp = p_data->write_req.need_rsp;
                (*p_inst->p_cback)(trans_id, conn_id, p_inst->app_id, p_inst->pending_evt, &cfg);
            } else { /* all other handle is not writable */
                st = ESP_GATT_WRITE_NOT_PERMIT;
            }
            break;
        }

    }
}
/***************************************************************
**
** Function     bas_register
**
** Description  register app for battery service
**
****************************************************************/
void bas_register(void)
{
    esp_ble_gatts_app_register(&bas_uuid, bas_gatts_callback);

}
/***************************************************************
**
** Function     bas_init
**
** Description  register battery service
**
****************************************************************/
void bas_init(tBTA_GATTS_IF gatt_if, UINT16 app_id)
{

    tBA_INST    *p_inst;

    ba_reg_info.is_pri = true;
    ba_reg_info.ba_level_descr = BA_LEVEL_NOTIFY;
    ba_reg_info.transport = GATT_TRANSPORT_LE;
    ba_reg_info.p_cback = bas_callback;
    if (battery_cb.inst_id == BA_MAX_INT_NUM) {
        GATT_TRACE_ERROR("MAX battery service has been reached\n");
        return;
    }

    p_inst = &battery_cb.battery_inst[battery_cb.inst_id];

    BTC_TRACE_ERROR("create battery service\n");
    BTC_TRACE_ERROR("inst_id=%d\n", battery_cb.inst_id);
    esp_ble_gatts_create_srvc (gatt_if, &bas_uuid, battery_cb.inst_id ,
                               BA_MAX_ATTR_NUM, ba_reg_info.is_pri);

    battery_cb.inst_id ++;

    p_inst->app_id  =   app_id;
    p_inst->p_cback =   ba_reg_info.p_cback;

}

/***************************************************************
**
** Function     bas_AddChar
**
** Description  add characteristic for battery service
**
****************************************************************/
void bas_AddChar(UINT16 service_id, tBT_UUID *char_uuid)
{
    if (ba_reg_info.ba_level_descr & BA_LEVEL_NOTIFY) {
        prop |= GATT_CHAR_PROP_BIT_NOTIFY;
    }
    attr_handle_bit = 0x01;
    esp_ble_gatts_add_char(service_id, char_uuid, BATTER_LEVEL_PERM, prop);

}

/***************************************************************
**
** Function     bas_AddCharDescr
**
** Description  add descriptor for battery service if needed
**
****************************************************************/
void bas_AddCharDescr(UINT16 service_id, UINT16 attr_id)
{
    tBT_UUID uuid;
    uuid.len = LEN_UUID_16;

    battery_cb.inst_id --;
    tBA_INST *p_inst = &battery_cb.battery_inst[battery_cb.inst_id++];
    /*store the attribute handles*/
    if (attr_handle_bit == 0x01) {
        p_inst->ba_level_hdl = attr_id;
    } else if (attr_handle_bit == 0x02) {
        p_inst->clt_cfg_hdl = attr_id;
    } else if (attr_handle_bit == 0x04) {
        p_inst->pres_fmt_hdl = attr_id;
    } else if (attr_handle_bit == 0x08) {
        p_inst->rpt_ref_hdl = attr_id;
    }


    if (ba_reg_info.ba_level_descr != 0) {
        if (ba_reg_info.ba_level_descr & BA_LEVEL_NOTIFY) {
            uuid.uu.uuid16 = GATT_UUID_CHAR_CLIENT_CONFIG;
            ba_reg_info.ba_level_descr &= 0xfe;
            attr_handle_bit = 0x02;
            esp_ble_gatts_add_char_descr(service_id, (GATT_PERM_READ | GATT_PERM_WRITE), &uuid);
            return;
        }

        /* need presentation format descriptor? */
        if (ba_reg_info.ba_level_descr & BA_LEVEL_PRE_FMT) {
            uuid.uu.uuid16 = GATT_UUID_CHAR_PRESENT_FORMAT;
            esp_ble_gatts_add_char_descr(service_id, GATT_PERM_READ, &uuid);
            ba_reg_info.ba_level_descr &= 0xfd;
            attr_handle_bit = 0x04;
            return;
        }
        /* need report reference format descriptor? */
        if (ba_reg_info.ba_level_descr & BA_LEVEL_RPT_REF) {
            uuid.uu.uuid16 = GATT_UUID_RPT_REF_DESCR;
            ba_reg_info.ba_level_descr &= 0xfb;
            esp_ble_gatts_add_char_descr(service_id, GATT_PERM_READ, &uuid);
            attr_handle_bit = 0x08;
            return;
        }
    }

    else {
        esp_ble_gatts_start_srvc(service_id);
    }

}

/***************************************************************
**
** Function     bas_service_cmpl
**
** Description  create battery service complete
**
****************************************************************/
void bas_service_cmpl(UINT16 service_id, esp_gatt_status_t status)
{
    if (status != ESP_GATT_OK) {
        battery_cb.inst_id --;
        esp_ble_gatts_dele_srvc(service_id);
    }

}
/*******************************************************************************
**
** Function         Battery_Rsp
**
** Description      Respond to a battery service request
**
*******************************************************************************/
void Battery_Rsp (UINT32 trans_id, UINT16 conn_id, UINT8 app_id,
                  esp_gatt_status_t st, UINT8 event, tBA_RSP_DATA *p_rsp)
{
    tBA_INST *p_inst = &battery_cb.battery_inst[0];
    tGATTS_RSP  rsp;
    UINT8   *pp;

    UINT8   i = 0;
    while (i < BA_MAX_INT_NUM) {
        if (p_inst->app_id == app_id && p_inst->ba_level_hdl != 0) {
            break;
        }
        i ++;
    }

    if (i == BA_MAX_INT_NUM) {
        return;
    }

    memset(&rsp, 0, sizeof(tGATTS_RSP));

    if (p_inst->pending_evt == event) {
        switch (event) {
        case BA_READ_CLT_CFG_REQ:
            rsp.attr_value.handle = p_inst->pending_handle;
            rsp.attr_value.len = 2;
            pp = rsp.attr_value.value;
            UINT16_TO_STREAM(pp, p_rsp->clt_cfg);
            esp_ble_gatts_send_rsp(conn_id, trans_id, st, &rsp);
            //srvc_sr_rsp(p_inst->pending_clcb_idx, st, &rsp);
            break;

        case BA_READ_LEVEL_REQ:
            rsp.attr_value.handle = p_inst->pending_handle;
            rsp.attr_value.len = 1;
            pp = rsp.attr_value.value;
            UINT8_TO_STREAM(pp, p_rsp->ba_level);
            esp_ble_gatts_send_rsp(conn_id, trans_id, st, &rsp);
            //srvc_sr_rsp(p_inst->pending_clcb_idx, st, &rsp);
            break;

        case BA_WRITE_CLT_CFG_REQ:
            esp_ble_gatts_send_rsp(conn_id, trans_id, st, NULL);
            //srvc_sr_rsp(p_inst->pending_clcb_idx, st, NULL);
            break;

        case BA_READ_RPT_REF_REQ:
            rsp.attr_value.handle = p_inst->pending_handle;
            rsp.attr_value.len = 2;
            pp = rsp.attr_value.value;
            UINT8_TO_STREAM(pp, p_rsp->rpt_ref.rpt_id);
            UINT8_TO_STREAM(pp, p_rsp->rpt_ref.rpt_type);
            esp_ble_gatts_send_rsp(conn_id, trans_id, st, &rsp);
            //srvc_sr_rsp(p_inst->pending_clcb_idx, st, &rsp);
            break;

        default:
            break;
        }
        // p_inst->pending_clcb_idx = 0;
        p_inst->pending_evt = 0;
        p_inst->pending_handle = 0;
    }
    return;
}
/*******************************************************************************
**
** Function         Battery_Notify
**
** Description      Send battery level notification
**
*******************************************************************************/
void Battery_Notify (UINT16 conn_id, UINT8 app_id, BD_ADDR remote_bda, UINT8 battery_level)
{
    tBA_INST *p_inst = &battery_cb.battery_inst[0];
    UINT8    i = 0;

    while (i < BA_MAX_INT_NUM) {
        if (p_inst->app_id == app_id && p_inst->ba_level_hdl != 0) {
            break;
        }
        i ++;
    }

    if (i == BA_MAX_INT_NUM || p_inst->clt_cfg_hdl == 0) {
        return;
    }
    esp_ble_gatts_hdl_val_indica(conn_id, p_inst->ba_level_hdl, 1, &battery_level, false);
    //srvc_sr_notify(remote_bda, p_inst->ba_level_hdl, 1, &battery_level);

}
#endif