#include <stdint.h>
#include <stdbool.h>
#include <string.h>
#include <esp_mac.h>
#include "include/ps3.h"
#include "include/ps3_int.h"

/********************************************************************************/
/*                              C O N S T A N T S                               */
/********************************************************************************/

static const uint8_t hid_cmd_payload_ps3_enable[] = { 0x42, 0x03, 0x00, 0x00 };
static const uint8_t hid_cmd_payload_led_arguments[] = { 0xff, 0x27, 0x10, 0x00, 0x32 };


/********************************************************************************/
/*                         L O C A L    V A R I A B L E S                       */
/********************************************************************************/

static ps3_connection_callback_t ps3_connection_cb = NULL;
static ps3_connection_object_callback_t ps3_connection_object_cb = NULL;
static void *ps3_connection_object = NULL;


static ps3_event_callback_t ps3_event_cb = NULL;
static ps3_event_object_callback_t ps3_event_object_cb = NULL;
static void *ps3_event_object = NULL;

static bool is_active = false;


/********************************************************************************/
/*                      P U B L I C    F U N C T I O N S                        */
/********************************************************************************/

/*******************************************************************************
**
** Function         ps3Init
**
** Description      This initializes the bluetooth services to listen
**                  for an incoming PS3 controller connection.
**
**
** Returns          void
**
*******************************************************************************/
void ps3Init()
{
    ps3_spp_init();
    ps3_l2cap_init_services();
}

/*******************************************************************************
**
** Function         ps3Deinit
**
** Description      This deinitializes the bluetooth services to stop
**                  listening for incoming connections.
**
**
** Returns          void
**
*******************************************************************************/
void ps3Deinit()
{
    ps3_l2cap_deinit_services();
    ps3_spp_deinit();
}


/*******************************************************************************
**
** Function         ps3IsConnected
**
** Description      This returns whether a PS3 controller is connected, based
**                  on whether a successful handshake has taken place.
**
**
** Returns          bool
**
*******************************************************************************/
bool ps3IsConnected()
{
    return is_active;
}


/*******************************************************************************
**
** Function         ps3Enable
**
** Description      This triggers the PS3 controller to start continually
**                  sending its data.
**
**
** Returns          void
**
*******************************************************************************/
void ps3Enable()
{
    uint16_t len = sizeof(hid_cmd_payload_ps3_enable);
    hid_cmd_t hid_cmd;

    hid_cmd.code = hid_cmd_code_set_report | hid_cmd_code_type_feature;
    hid_cmd.identifier = hid_cmd_identifier_ps3_enable;

    memcpy( hid_cmd.data, hid_cmd_payload_ps3_enable, len);

    ps3_l2cap_send_hid( &hid_cmd, len );
}

/*******************************************************************************
**
** Function         ps3Cmd
**
** Description      Send a command to the PS3 controller.
**
**
** Returns          void
**
*******************************************************************************/
void ps3Cmd( ps3_cmd_t cmd )
{
    hid_cmd_t hid_cmd = { .data = {0} };
    uint16_t len = sizeof(hid_cmd.data);

    hid_cmd.code = hid_cmd_code_set_report | hid_cmd_code_type_output;
    hid_cmd.identifier = hid_cmd_identifier_ps3_control;

    hid_cmd.data[ps3_control_packet_index_rumble_right_duration]  = cmd.rumble_right_duration;
    hid_cmd.data[ps3_control_packet_index_rumble_right_intensity] = cmd.rumble_right_intensity;
    hid_cmd.data[ps3_control_packet_index_rumble_left_duration]   = cmd.rumble_left_duration;
    hid_cmd.data[ps3_control_packet_index_rumble_left_intensity]  = cmd.rumble_left_intensity;

    hid_cmd.data[ps3_control_packet_index_leds] = 0;
    if (cmd.led1) hid_cmd.data[ps3_control_packet_index_leds] |= ps3_led_mask_led1;
    if (cmd.led2) hid_cmd.data[ps3_control_packet_index_leds] |= ps3_led_mask_led2;
    if (cmd.led3) hid_cmd.data[ps3_control_packet_index_leds] |= ps3_led_mask_led3;
    if (cmd.led4) hid_cmd.data[ps3_control_packet_index_leds] |= ps3_led_mask_led4;

    if (cmd.led1) memcpy( hid_cmd.data + ps3_control_packet_index_led1_arguments, hid_cmd_payload_led_arguments, sizeof(hid_cmd_payload_led_arguments));
    if (cmd.led2) memcpy( hid_cmd.data + ps3_control_packet_index_led2_arguments, hid_cmd_payload_led_arguments, sizeof(hid_cmd_payload_led_arguments));
    if (cmd.led3) memcpy( hid_cmd.data + ps3_control_packet_index_led3_arguments, hid_cmd_payload_led_arguments, sizeof(hid_cmd_payload_led_arguments));
    if (cmd.led4) memcpy( hid_cmd.data + ps3_control_packet_index_led4_arguments, hid_cmd_payload_led_arguments, sizeof(hid_cmd_payload_led_arguments));

    ps3_l2cap_send_hid( &hid_cmd, len );
}


/*******************************************************************************
**
** Function         ps3SetLed
**
** Description      Sets the LEDs on the PS3 controller to the player
**                  number. Up to 10 players are supported.
**
**
** Returns          void
**
*******************************************************************************/
void ps3SetLed( uint8_t player )
{
    ps3_cmd_t cmd = {0};
    ps3SetLedCmd(&cmd, player);
    ps3Cmd(cmd);
}


/*******************************************************************************
**
** Function         ps3SetLed
**
** Description      Sets the LED bits on the PS3 controller command to the
**                  player number. Up to 10 players are supported.
**
**
** Returns          void
**
*******************************************************************************/
void ps3SetLedCmd( ps3_cmd_t *cmd, uint8_t player )
{

    //           led4  led3  led2  led1
    // player 1                    1
    // player 2              1
    // player 3        1
    // player 4  1
    // player 5  1                 1
    // player 6  1           1
    // player 7  1     1
    // player 8  1     1           1
    // player 9  1     1     1
    // player 10 1     1     1     1

    if( (cmd->led4 = player >= 4) != 0 ) player -= 4;
    if( (cmd->led3 = player >= 3) != 0 ) player -= 3;
    if( (cmd->led2 = player >= 2) != 0 ) player -= 2;
    if( (cmd->led1 = player >= 1) != 0 ) player -= 1;

}


/*******************************************************************************
**
** Function         ps3SetConnectionCallback
**
** Description      Registers a callback for receiving PS3 controller
**                  connection notifications
**
**
** Returns          void
**
*******************************************************************************/
void ps3SetConnectionCallback( ps3_connection_callback_t cb )
{
    ps3_connection_cb = cb;
}


/*******************************************************************************
**
** Function         ps3SetConnectionObjectCallback
**
** Description      Registers a callback for receiving PS3 controller
**                  connection notifications
**
**
** Returns          void
**
*******************************************************************************/
void ps3SetConnectionObjectCallback( void *object, ps3_connection_object_callback_t cb )
{
    ps3_connection_object_cb = cb;
    ps3_connection_object = object;
}

/*******************************************************************************
**
** Function         ps3SetEventCallback
**
** Description      Registers a callback for receiving PS3 controller events
**
**
** Returns          void
**
*******************************************************************************/
void ps3SetEventCallback( ps3_event_callback_t cb )
{
    ps3_event_cb = cb;
}


/*******************************************************************************
**
** Function         ps3SetEventObjectCallback
**
** Description      Registers a callback for receiving PS3 controller events
**
**
** Returns          void
**
*******************************************************************************/
void ps3SetEventObjectCallback( void *object, ps3_event_object_callback_t cb )
{
    ps3_event_object_cb = cb;
    ps3_event_object = object;
}


/*******************************************************************************
**
** Function         ps3SetBluetoothMacAddress
**
** Description      Writes a Registers a callback for receiving PS3 controller events
**
**
** Returns          void
**
*******************************************************************************/
void ps3SetBluetoothMacAddress( const uint8_t *mac )
{
    esp_iface_mac_addr_set(mac, ESP_MAC_BT);
}


/********************************************************************************/
/*                      L O C A L    F U N C T I O N S                          */
/********************************************************************************/

void ps3_connect_event( uint8_t is_connected )
{
    if(is_connected){
        ps3Enable();
    }else{
        is_active = false;
    }
}


void ps3_packet_event( ps3_t ps3, ps3_event_t event )
{
    // Trigger packet event, but if this is the very first packet
    // after connecting, trigger a connection event instead
    if(is_active){
        if(ps3_event_cb != NULL)
        {
            ps3_event_cb( ps3, event );
        }

        if(ps3_event_object_cb != NULL && ps3_event_object != NULL)
        {
            ps3_event_object_cb( ps3_event_object, ps3, event );
        }
    }else{
        is_active = true;

        if(ps3_connection_cb != NULL)
        {
            ps3_connection_cb( is_active );
        }

        if(ps3_connection_object_cb != NULL && ps3_connection_object != NULL)
        {
            ps3_connection_object_cb( ps3_connection_object, is_active );
        }
    }
}
