/*====================================================================*
	Copyright (c) 2013-2018 Qualcomm Technologies, Inc.
	All Rights Reserved.
	Confidential and Proprietary - Qualcomm Technologies, Inc.
	******************************************************************
	2013 Qualcomm Atheros, Inc.
 *--------------------------------------------------------------------*/

/*====================================================================*
 *
 *   signed Attributes2 (struct plc * plc);
 *
 *   plc.h
 *
 *   This plugin for program plc requests device attributes using
 *   a VS_OP_ATTRIBUTES message; attributes are pre-parsed versions
 *   of information returned by VS_SW_VER and other messages;
 *
 *   The VS_OP_ATTRIBUTES message structure changed between FW 3.3.4
 *   and 3.3.5 and fields are ocassionally appended to the end; you
 *   should not use this message for operational systems because the
 *   format may change again; it was originally intended for PTS use
 *   only;
 *
 *.  Qualcomm Atheros HomePlug AV Powerline Toolkit
 *:  Published 2009-2012, Qualcomm Atheros. ALL RIGHTS RESERVED
 *;  For demonstration and evaluation only. Not for production use
 *
 *   Contributor(s):
 *      Charles Maier <cmaier@qca.qualcomm.com>
 *
 *--------------------------------------------------------------------*/

#ifndef ATTRIBUTES2_SOURCE
#define ATTRIBUTES2_SOURCE

#include <stdint.h>
#include <memory.h>

#include "../plc/plc.h"
#include "../tools/memory.h"
#include "../tools/format.h"
#include "../tools/error.h"
#include "../tools/flags.h"

#ifdef WIN32
#define snprintf _snprintf
#endif

signed Attributes2 (struct plc * plc)

{
	struct channel * channel = (struct channel *) (plc->channel);
	struct message * message = (struct message *) (plc->message);

#ifndef __GNUC__
#pragma pack (push,1)
#endif

	struct __packed vs_op_attributes_request
	{
		struct ethernet_hdr ethernet;
		struct qualcomm_hdr qualcomm;
		uint32_t COOKIE;
		uint8_t RTYPE;
	}
	* request = (struct vs_op_attributes_request *) (message);
	struct __packed vs_op_attributes_confirm
	{
		struct ethernet_hdr ethernet;
		struct qualcomm_hdr qualcomm;
		uint16_t MSTATUS;
		uint32_t COOKIE;
		uint8_t RTYPE;
		uint16_t MLENGTH;
		uint8_t MBUFFER [1024];
	}
	* confirm = (struct vs_op_attributes_confirm *) (message);
	struct __packed attributes
	{
		uint8_t DEVICEFAMILY [16];
		uint8_t DEVICETYPE [16];
		uint32_t FWMAJORVERSION;
		uint32_t FWMINORVERSION;
		uint32_t COMPOSITEVERSION;
		uint32_t SUSTAININGVERSION;
		uint32_t BUILDNUMBER;
		uint8_t BUILDDATE [8];
		uint8_t RELEASETYPE [12];
		uint8_t FLAGS [3];
		uint32_t DRAMSIZE;
		uint32_t RAMBLOCKRXCOUNT;
		uint32_t RAMBLOCKTXCOUNT;
		uint32_t RAMBLOCKSHAREDCOUNT;
		uint32_t RAMBLOCKFREERXCOUNT;
		uint32_t RAMBLOCKFREETXCOUNT;
		uint32_t RAMBLOCKFREESHAREDCOUNT;
		uint32_t RELATIVESNRDIFFDB;
		uint32_t DSP384THREGVAL;
		uint8_t AFETXGAINDB;
		uint8_t AUTHMODE;
		uint8_t MICROCONTROLLERDIAGENABLED;
		uint8_t ADAPTERTYPE;
		uint8_t EN50561MODE;
		uint16_t EN50561TONE;
		uint32_t DEVICEUPTIMESINCELASTWARMRESET;
		uint32_t DEVICEUPTIMESINCELASTHARDRESET;
	}
	* attributes = (struct attributes *) (confirm->MBUFFER);

#ifndef __GNUC__
#pragma pack (pop)
#endif

	Request (plc, "Fetch Device Attributes");
	memset (message, 0, sizeof (* message));
	EthernetHeader (& request->ethernet, channel->peer, channel->host, channel->type);
	QualcommHeader (& request->qualcomm, 0, (VS_OP_ATTRIBUTES | MMTYPE_REQ));
	plc->packetsize = (ETHER_MIN_LEN - ETHER_CRC_LEN);
	request->COOKIE = HTOLE32 (plc->cookie);
	request->RTYPE = 0;
	if (SendMME (plc) <= 0)
	{
		error (PLC_EXIT (plc), errno, CHANNEL_CANTSEND);
		return (-1);
	}
	while (ReadMME (plc, 0, (VS_OP_ATTRIBUTES | MMTYPE_CNF)) > 0)
	{

#if 0

		static char const * frequency [] =
		{
			"Unknown Frequency",
			"50Hz",
			"60Hz",
		};
		static char const * zero_cross [] =
		{
			"Not yet detected",
			"Detected",
			"Missing",
		};

#endif
#if 1

		static char const * adapt_type [] =
		{
			"Unknown",
			"Wall Adapter",
			"SmartLink",
			"SmartLink Plus",
		};
		static char const * en_mode [] =
		{
			"EN50561-1 Not detected",
			"EN50561-1 Static mode",
			"EN50561-1 Dynamic mode",
		};

#endif

		char string [512];
		size_t length = 0;
		if (confirm->MSTATUS)
		{
			Failure (plc, PLC_WONTDOIT);
			continue;
		}

#if 0 
#if defined (__GNUC__)
#warning "Debug code active in module Attributes2" 
#endif

		fprintf (stderr, "DEVICE FAMILY=[%-16.16s]\n", attributes->DEVICEFAMILY);
		fprintf (stderr, "DEVICE TYPE=[%-16.16s]\n", attributes->DEVICETYPE);
		fprintf (stderr, "FIRMWARE MAJOR VERSION=(%d)\n", LE32TOH (attributes->FWMAJORVERSION));
		fprintf (stderr, "FIRMWARE MINOR VERSION=(%d)\n", LE32TOH (attributes->FWMINORVERSION));
		fprintf (stderr, "COMPOSITE VERSION=(%d)\n", LE32TOH (attributes->COMPOSITEVERSION));
		fprintf (stderr, "SUSTAINING VERSION=(%d)\n", LE32TOH (attributes->SUSTAININGVERSION));
		fprintf (stderr, "BUILD NUMBER=(%d)\n", LE32TOH (attributes->BUILDNUMBER));
		fprintf (stderr, "BUILD DATE=[%-8.8s]\n", attributes->BUILDDATE);
		fprintf (stderr, "RELEASE TYPE=[%-12.12s]\n", attributes->RELEASETYPE);
		fprintf (stderr, "DRAM SIZE=[0x%08X](%d)\n", LE32TOH (attributes->DRAMSIZE), LE32TOH (attributes->DRAMSIZE));
		fprintf (stderr, "DRAM TYPES=[0x%02X]\n", attributes->FLAGS [0]);
		fprintf (stderr, "RESERVED FLAG=[0x%02X]\n", attributes->FLAGS [1]);
		fprintf (stderr, "LINE FREQUENCY=[0x%02X]\n", attributes->FLAGS [2]);
		fprintf (stderr, "AUTHORIZATION MODE=[0x%02X]\n", attributes->AUTHMODE);
		fprintf (stderr, "DSP384THREGVAL=[0x%02X]\n", attributes->DSP384THREGVAL);
		fprintf (stderr, "AFETXGAINDB=[0x%02X]\n", attributes->AFETXGAINDB);
		fprintf (stderr, "AUTHORIZATION MODE=[0x%02X]\n", attributes->AUTHMODE);
		fprintf (stderr, "ADAPTER TYPE=[0x%02X]\n", attributes->ADAPTERTYPE);
		fprintf (stderr, "EN50561 MODE=[0x%02X]\n", attributes->EN50561MODE);
		fprintf (stderr, "EN50561 TONES=[0x%02X]\n", attributes->EN50561TONE);

#if 0

		fprintf (stderr, "DSP384THREGVAL=[0x%08X]\n", LE32TOH (attributes->DSP384THREGVAL));
		fprintf (stderr, "AFETXGAINDB=[0x%02X]\n", attributes->AFETXGAINDB);

#endif
#endif

		length += snprintf (string +  length, sizeof (string) - length, "%s", attributes->DEVICEFAMILY);
		length += snprintf (string +  length, sizeof (string) - length, "-%s", attributes->DEVICETYPE);
		length += snprintf (string +  length, sizeof (string) - length, "-%s", attributes->DEVICEFAMILY);
		length += snprintf (string +  length, sizeof (string) - length, "-%d", LE32TOH (attributes->FWMAJORVERSION));
		length += snprintf (string +  length, sizeof (string) - length, ".%d", LE32TOH (attributes->FWMINORVERSION));
		length += snprintf (string +  length, sizeof (string) - length, ".%d", LE32TOH (attributes->COMPOSITEVERSION));
		length += snprintf (string +  length, sizeof (string) - length, ".%d", LE32TOH (attributes->BUILDNUMBER));
		length += snprintf (string +  length, sizeof (string) - length, "-%02d", LE32TOH (attributes->SUSTAININGVERSION));
		length += snprintf (string +  length, sizeof (string) - length, "-%8.8s", attributes->BUILDDATE);
		length += snprintf (string +  length, sizeof (string) - length, "-%s", attributes->RELEASETYPE);
		length += snprintf (string +  length, sizeof (string) - length, " (%dmb)", LE32TOH (attributes->DRAMSIZE));
		length += snprintf (string +  length, sizeof (string) - length, " (%s)", adapt_type [attributes->ADAPTERTYPE]);
		length += snprintf (string +  length, sizeof (string) - length, " (%s)", en_mode [attributes->EN50561MODE]);
		length += snprintf (string +  length, sizeof (string) - length, " (%d)", LE32TOH (attributes->EN50561TONE));
		length += snprintf (string +  length, sizeof (string) - length, " Uptime since last Warm Reset(%d) secs", LE32TOH (attributes->DEVICEUPTIMESINCELASTWARMRESET));
		length += snprintf (string +  length, sizeof (string) - length, " Hard Reset(%d) secs", LE32TOH (attributes->DEVICEUPTIMESINCELASTHARDRESET));

/*
 *   length += snprintf (string +  length, sizeof (string) - length, " (0x%02X) \"%s\" \"%s\" ", attributes->FLAGS, frequency [attributes->FLAGS [1] & 0x03], zero_cross [(attributes->FLAGS [2] & 0x0C) >> 2]);
 */

		Display (plc, "%s", string);
	}
	return (0);
}



#endif