/*******************************************************************************
* PMC825 UDP/IP Socket Interface Test Program                                  *
*                                                                              *
* (C) 2010 Stock Flight Systems. All rights reserved.                          *
*                                                                              *
* Filename: test_device_1.c                                                    *
*                                                                              *
* 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.                       *
*                                                                              *
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER "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 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.                           *
*                                                                              *
* This file contains a test program which interfaces to the PMC825 via the     *
* UDP/IP socket interface.                                                     *
*                                                                              *
* Function names                                                               *
* ____________________________________________________________________________ *
*                                                                              *
* int main(void)                                                               *
* void ExceptionHandler(void)                                                  *
*                                                                              *
* MODIFICATIONS:                                                               *
*                                                                              *
* When          Version      What                                  Who         *
* ____________________________________________________________________________ *
*                                                                              *
* 22.04.2010    1.0          Initial Version                       M. Stock    *
* 11.06.2010    1.1          Some minor bugs fixed                 M. Stock    *
*                                                                              *
*******************************************************************************/

/*
 * Includes.
 */

#include <math.h>
#include "pmc825socket.h"
#include "arinc825.h"

/*
 * Local definitions.
 */

#define	LPORT_BASE	34567           /* Local UDP/IP port number base */
#define	RPORT_BASE      34568           /* PMC825 UDP/IP port number base */

/*
 * Globals.
 */

PMC825_IF Pmc825;

/*
 * ExceptionHandler() frees all resources then terminates.
 */

void ExceptionHandler(void)
{
CTRL_MSG tx_ctrl;
int ret;

/*
 * Stop the CPM.
 */

tx_ctrl.opcode = CAN_CTRL;
tx_ctrl.svc_rsp_code = INIT_CAN_CHIP;
tx_ctrl.arg[0] = CAN_1M;
tx_ctrl.arg[1] = 0;
tx_ctrl.arg[2] = 0;
tx_ctrl.arg[3] = 0;
tx_ctrl.arg[4] = 0;

ret = Pmc825CtrlWrite(&Pmc825, &tx_ctrl);

if (ret < 0)
  printf("Pmc825CtrlWrite[INIT_CAN_CHIP]: %d\n", ret);

Pmc825StopInterface(&Pmc825);
printf("\nProgram terminated, all resources released.\n");
exit(0);
}

/*
 * main() starts here.
 */

int main (int argc, char *argv[])
{
#ifndef WIN32
struct timespec t2,t1 = {0,1000000*5};	/* 10ms frame time */
#endif
unsigned short tmr_res;
unsigned int *iptr, host_ip, pmc825_ip, lport, rport;
unsigned char tmp[4], hip[32], pip[32], ch[8];
int loops, ctrl_seq, statseq, ret, chan, ip[4];
ARINC825_MSG tx_buf[12];
CTRL_MSG tx_ctrl, rx_ctrl;
CAN_STAT_REGS *stat;

/*
 * First of all, get IP addresses and CAN channel number.
 */

if (argc != 4)
  {
  printf("usage: %s host_ip pmc825_ip can_channel\n", argv[0]);
  exit(1);
  }

strcpy(hip, argv[1]);
strcpy(pip, argv[2]);
strcpy(ch, argv[3]);

sscanf(hip, "%d.%d.%d.%d", &ip[0], &ip[1], &ip[2], &ip[3]);
host_ip = (ip[0] << 24) | (ip[1] << 16) | (ip[2] << 8) | ip[3];

sscanf(pip, "%d.%d.%d.%d", &ip[0], &ip[1], &ip[2], &ip[3]);
pmc825_ip = (ip[0] << 24) | (ip[1] << 16) | (ip[2] << 8) | ip[3];

sscanf(ch, "%d", &chan);

lport = LPORT_BASE + chan * 2;
rport = RPORT_BASE + chan * 2;

tmr_res = 1 << chan;

/*
 * Set up the exception handler. We want to make sure that we perform
 * cleanup in case we are terminated by some unexpected event.
 */

#ifndef WIN32
signal(SIGQUIT, (void *) ExceptionHandler);
signal(SIGINT,  (void *) ExceptionHandler);
signal(SIGTERM, (void *) ExceptionHandler);
signal(SIGKILL, (void *) ExceptionHandler);
signal(SIGPIPE, (void *) ExceptionHandler);
#endif

/*
 * Initialize the interface for the PMC825.
 */

ret = Pmc825StartInterface(&Pmc825, pmc825_ip, host_ip, rport, lport, chan);

if (ret == PMC825_MEM_ALLOC_ERR)
  {
  printf("Memory allocation error, exiting ...\n");
  Pmc825StopInterface(&Pmc825);
  exit(0);
  }
else if (ret == PMC825_SOCKET_ERR)
  {
  printf("Socket error, exiting ...\n");
  Pmc825StopInterface(&Pmc825);
  exit(0);
  }

/*
 * Setup the ARINC825 message buffer.
 */

for (loops = 0; loops < 12; loops++)
  {
  tx_buf[loops].identifier.lcc = NOC;
  tx_buf[loops].identifier.scfid = 5;
  tx_buf[loops].identifier.smt = 0;
  tx_buf[loops].identifier.lcl = 0;
  tx_buf[loops].identifier.pvt = 1;
  tx_buf[loops].identifier.rci = RCI_A;
  tx_buf[loops].frame_type = DATA_FRAME;
  tx_buf[loops].can_status = 0;
  tx_buf[loops].error_counter = 0;
  tx_buf[loops].time_stamp_hi = 0;

  iptr = (unsigned int *) &(tx_buf[loops].data[0]);
  *iptr = 0;

  iptr = (unsigned int *) &(tx_buf[loops].data[4]);
  *iptr = 0;
  }

tx_buf[0].identifier.doc = 50;
tx_buf[0].byte_count = 5;
tx_buf[0].time_stamp_lo = CPM_TIME_1MS;
tx_buf[0].data[0] = 50;

tx_buf[1].identifier.doc = 60;
tx_buf[1].byte_count = 6;
tx_buf[1].time_stamp_lo = tx_buf[0].time_stamp_lo + CPM_TIME_500US;
tx_buf[1].data[0] = 60;

tx_buf[2].identifier.doc = 70;
tx_buf[2].byte_count = 7;
tx_buf[2].time_stamp_lo = tx_buf[1].time_stamp_lo + CPM_TIME_500US;
tx_buf[2].data[0] = 70;

tx_buf[3].identifier.doc = 80;
tx_buf[3].byte_count = 8;
tx_buf[3].time_stamp_lo = tx_buf[2].time_stamp_lo + CPM_TIME_500US;
tx_buf[3].data[0] = 80;

tx_buf[4].identifier.doc = 10;
tx_buf[4].byte_count = 1;
tx_buf[4].time_stamp_lo = tx_buf[3].time_stamp_lo + CPM_TIME_500US;
tx_buf[4].data[0] = 10;

tx_buf[5].identifier.doc = 20;
tx_buf[5].byte_count = 2;
tx_buf[5].time_stamp_lo = tx_buf[4].time_stamp_lo + CPM_TIME_500US;
tx_buf[5].data[0] = 20;

tx_buf[6].identifier.doc = 30;
tx_buf[6].byte_count = 3;
tx_buf[6].time_stamp_lo = tx_buf[5].time_stamp_lo + CPM_TIME_500US;
tx_buf[6].data[0] = 30;

tx_buf[7].identifier.doc = 40;
tx_buf[7].byte_count = 4;
tx_buf[7].time_stamp_lo = tx_buf[6].time_stamp_lo + CPM_TIME_500US;
tx_buf[7].data[0] = 40;

tx_buf[8].identifier.doc = 50;
tx_buf[8].byte_count = 5;
tx_buf[8].time_stamp_lo = tx_buf[7].time_stamp_lo + CPM_TIME_500US;
tx_buf[8].data[0] = 50;

tx_buf[9].identifier.doc = 60;
tx_buf[9].byte_count = 6;
tx_buf[9].time_stamp_lo = tx_buf[8].time_stamp_lo + CPM_TIME_500US;
tx_buf[9].data[0] = 60;

tx_buf[10].identifier.doc = 70;
tx_buf[10].byte_count = 7;
tx_buf[10].time_stamp_lo = tx_buf[9].time_stamp_lo + CPM_TIME_500US;
tx_buf[10].data[0] = 70;

tx_buf[11].identifier.doc = 80;
tx_buf[11].byte_count = 8;
tx_buf[11].time_stamp_lo = tx_buf[0].time_stamp_lo + CPM_TIME_1MS * 9;
tx_buf[11].data[0] = 80;

/*
 * Initialize other variables.
 */

ctrl_seq = -1;
statseq = -1;

/*
 * Now go into an endless loop controlling the PMC825.
 */

printf("Start Device Simulation on CAN channel %d:\n", chan);

for (;;)
  {
  /*
   * Set up the device generator and start it.
   */

  if (ctrl_seq < 2)
    ctrl_seq += 1;

  switch (ctrl_seq)
    {
    case 0:
      ret = Pmc825Arinc825Write(&Pmc825, &(tx_buf[0]), 12);

      if (ret < 0)
        printf("Pmc825CanAerospaceWrite[A]: %d\n", ret);
      break;

    case 1:
      tx_ctrl.opcode = CAN_CTRL;
      tx_ctrl.svc_rsp_code = INIT_CAN_CHIP;
      tx_ctrl.arg[0] = CAN_1M;
      tx_ctrl.arg[1] = 0;
      tx_ctrl.arg[2] = 0;
      tx_ctrl.arg[3] = 0;
      tx_ctrl.arg[4] = CPM_ENABLE | CPM_CYCLIC;

      ret = Pmc825CtrlWrite(&Pmc825, &tx_ctrl);

      if (ret < 0)
	printf("Pmc825CtrlWrite[INIT_CAN_CHIP]: %d\n", ret);
      break;

    case 2:
      tx_ctrl.opcode = CAN_CTRL;
      tx_ctrl.svc_rsp_code = RESET_TIME_STAMP;
      tx_ctrl.arg[0] = tmr_res;

      Pmc825CtrlWrite(&Pmc825, &tx_ctrl);
      break;

    default:
      break;
    }

  /*
   * Get received CAN control messages and print them.
   */

  ret = Pmc825CtrlRead(&Pmc825, &rx_ctrl);

  while (ret != PMC825_NO_MSG)
    {
    switch (rx_ctrl.opcode)
      {
      case CAN_STATUS:
        if (statseq < 50)
          statseq++;
        else
          {
          statseq = 0;
          stat = (CAN_STAT_REGS *) &(rx_ctrl.arg[0]);

          printf("\nEthernet STATUS Packet\n");
          printf("----------------------\n");
          printf("PMC825 Name       = %s\n",stat->name);
          printf("PMC825 IP Address = %d.%d.%d.%d\n",stat->ip_addr[0],
                                                     stat->ip_addr[1],
                                                     stat->ip_addr[2],
                                                     stat->ip_addr[3]);
          printf("CAN Channel       = %d\n",chan);
          printf("Bus Timing Reg    = 0x%04x\n",stat->btr);
          printf("CAN Status        = 0x%04x\n",stat->can_status);
          printf("Error Counters    = 0x%04x\n",stat->error_counter);
          printf("CPM Setting       = 0x%04x\n",stat->cpm_mode);
          printf("TX Msgs (100ms)   = %d\n",    stat->tx_msgs);
          printf("RX Msgs (100ms)   = %d\n",    stat->rx_msgs);
          printf("TX Bits (100ms)   = %d\n",    stat->tx_bits);
          printf("RX Bits (100ms)   = %d\n",    stat->rx_bits);
          printf("Board Temperature = %d\n",    stat->t1);
          printf("FPGA Temperature  = %d\n",    stat->t2);
          sleep(1);
          }
        break;

      default:
        break;
      }
    ret = Pmc825CtrlRead(&Pmc825, &rx_ctrl);
    }

  /*
   * 10ms time frame.
   */

#ifndef WIN32
  nanosleep(&t1,&t2);
#else
  Sleep (10);
#endif
  }
exit(0);
}

/*
 * End of file.
 */
