// Copyright (c) 2013, Jan Winkler <winkler@cs.uni-bremen.de>
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
//     * Redistributions of source code must retain the above copyright
//       notice, this list of conditions and the following disclaimer.
//     * Redistributions in binary form must reproduce the above copyright
//       notice, this list of conditions and the following disclaimer in the
//       documentation and/or other materials provided with the distribution.
//     * Neither the name of Universität Bremen nor the names of its
//       contributors may be used to endorse or promote products derived from
//       this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.


/* \author Jan Winkler */


#ifndef __C_CRAZY_RADIO_H__
#define __C_CRAZY_RADIO_H__

#define ALL_THE_DEBUG 0

// System
#include <list>
#include <string>
#include <cstdio>
#include <cstring>
#include <libusb-1.0/libusb.h>
#include <unistd.h>
#include <iostream>
#include <sstream>
#include <vector>

// Private
#include "structures.h"
#include "CCRTPPacket.h"
#include "CCrazyflie.h"

// Backward declaration of the CCrazyflie class to make the compiler happy
class CCrazyflie;

/*! \brief Communication class to connect to and communicate via the
    CrazyRadio USB dongle.

    The class is capable of finding the CrazyRadio USB dongle on the
    host computer, open and maintain a connection, and send/receive
    data when communicating with the Crazyflie Nano copter using the
    Crazy Radio Transfer Protocol as defined by Bitcraze. */
class CCrazyRadio {
private:
  // Variables
  /*! \brief The radio URI as supplied when initializing the class
      instance */
  //std::string m_strRadioIdentifier;  -- removing this, as we want our radio to support more than one crazyflie
  /*! \brief The current USB context as supplied by libusb */
  libusb_context *m_ctxContext;
  libusb_device *m_devDevice;
  libusb_device_handle *m_hndlDevice;
  int m_nARC;
  //int m_nChannel;             -- moving this out to the crazyflie
  //std::string m_strDataRate;  -- so that multiple flies can be on the same radio.
  int m_nARDTime;
  int m_nARDBytes;
  int m_nDongleNumber;
  enum Power m_enumPower;
  char *m_cAddress;
  int m_bContCarrier;
  float m_fDeviceVersion;
  bool m_bAckReceived;

  bool m_exitThread;

  // Vector to hold the crazyflie objects associated with this radio
  std::vector<CCrazyflie*> crazyflies;

  // Functions
  std::list<libusb_device*> listDevices(int nVendorID, int nProductID);
  bool openUSBDongle();
  bool claimInterface(int nInterface);
  void closeDevice();

 

  CCRTPPacket *writeData(void *vdData, int nLength);
  bool writeControl(void *vdData, int nLength, uint8_t u8Request, uint16_t u16Value, uint16_t u16Index);
  bool readData(void *vdData, int &nMaxLength);

  void setAddress(char *cAddress);
  void setContCarrier(bool bContCarrier);

public:

bool isLogging = false;

  /**
   * Add a Crazyflie to the radio
   *
   * @param flie Pointer to the CCrazyflie object
   */
  void addCrazyflie(CCrazyflie *flie);

   CCRTPPacket *readACK();

  /**
   * Main loop for the thread running the radio
   */
  void mainLoop();

  /**
   * Static function to use to start the thread for the radio
   */
  static void* startThread(void* args);
  void stopThread();

	void setDataRate(XferRate rate);  //Moved From Private

  void setARC(int nARC);		//Moved From Private

  void setARDBytes(int nARDBytes);	//Moved From Private
  void setARDTime(int nARDTime);		//Moved From Private

  void setChannel(int nChannel);
  /*! \brief Constructor for the radio communication class

    \param dongleNumber 0-based index of the USB dongle to use. */
  //CCrazyRadio(std::string strRadioIdentifier);  -- removing for multiple crazyflie support
  CCrazyRadio(int dongleNumber);
  /*! \brief Destructor for the radio communication class */
  ~CCrazyRadio();

  /*! \brief Function to start the radio communication

    The first available USB dongle will be opened and claimed for
    communication. The connection will be maintained and used to
    communicate with a Crazyflie Nano quadcopter in range.

    \return Returns 'true' if the connection could successfully be
    made and 'false' if no dongle could be found (or any other
    USB-related error came up - this is not handled here). */
  bool startRadio();

  /*! \brief Returns the current setting for power usage by the USB
      dongle

    \return Value denoting the current power settings reserved for
    communication */
  enum Power power();
  /*! \brief Set the power level to be used for communication purposes

    \param enumPower The level of power that is being used for
    communication. The integer value maps to one of the entries of the
    Power enum. */
  void setPower(enum Power enumPower);

  /*! \brief Sends the given packet's payload to the copter

    \param crtpSend The packet which supplied header and payload
    information to send to the copter */
  CCRTPPacket *sendPacket(int nRadioChannel, CCRTPPacket *crtpSend, CCrazyflie *crazyflie);

  /*! \brief Sends the given packet and waits for a reply.

    Internally, this function calls the more elaborate
    sendAndReceive() function supplying parameters for retrying and
    waiting. Convenience function signature.

    \param crtpSend Packet to send
    \param bDeleteAfterwards Whether or not the packet to send is
    deleted internally after sending it

    \return Packet containing the reply or NULL if no reply was
    received (after retrying). */
  CCRTPPacket *sendAndReceive(CCRTPPacket *crtpSend, int nRadioChannel, CCrazyflie *crazyflie);

  /*! \brief Sends the given packet and waits for a reply.

    Sends out the CCRTPPacket instance denoted by crtpSend on the
    given port and channel. Retries a number of times and waits
    between each retry whether or not an answer was received (in this
    case, dummy packets are sent in order to receive replies).

    \param crtpSend Packet to send

    \param nPort Port number on which to send this packet (and where
    to wait for the reply)
    \param nChannel Channel number on which to send this packet (and
    where to wait for the reply)
    \param bDeletAfterwards Whether or not the packet to send is
    deleted internally after sending it
    \param nRetries Number of retries (re-sending) before giving up on
    an answer
    \param nMicrosecondsWait Microseconds to wait between two re-sends

    \return Packet containing the reply or NULL if no reply was
    received (after retrying). */
  CCRTPPacket *sendAndReceive(CCRTPPacket *crtpSend, int nRadioChannel, int nPort, int nChannel, CCrazyflie *crazyflie, int nRetries = 10, int nMicrosecondsWait = 100);				//TODO ******CHANGED nRetries from 10 to 1 - JUSTIN*******

  /*! \brief Sends out an empty dummy packet

    Only contains the payload `0xff`, as used for empty packet
    requests. Mostly used for waiting or keepalive.

    \return Boolean value denoting whether sending the dummy packet
    worked or not. */
  bool sendDummyPacket(int nChannel, CCrazyflie *crazyflie);

  /*! \brief Waits for the next non-empty packet.

    Sends out dummy packets until a reply is non-empty and then
    returns this reply.

    \return Packet contaning a non-empty reply. */
  CCRTPPacket *waitForPacket( int nChannel, CCrazyflie *crazyflie);

  /*! \brief Whether or not the copter is answering sent packets.

    Returns whether the copter is actually answering sent packets with
    a set ACK flag. If this is not the case, it is either switched off
    or out of range.

    \return Returns true if the copter is returning the ACK flag properly, false otherwise. */
  bool ackReceived();
  /*! \brief Whether or not the USB connection is still operational.

    Checks if the USB read/write calls yielded any errors.

    \return Returns true if the connection is working properly and
    false otherwise. */
  bool usbOK();

  /*! \brief Extracting all logging related packets

    Returns a list of all collected logging related (i.e. originating
    from port 5) packets. This is called by the CCrazyflie class
    automatically when performing cycle().

    \return List of CCRTPPacket instances collected from port 5
    (logging). */
//  std::list<CCRTPPacket*> popLoggingPackets();
//  void clearPackets();

  bool bPowerDetector;
  int nRetransmissions;

};


#endif /* __C_CRAZY_RADIO_H__ */