/**
* Copyright 2005-2007 ECMWF
*
* Licensed under the GNU Lesser General Public License which
* incorporates the terms and conditions of version 3 of the GNU
* General Public License.
* See LICENSE and gpl-3.0.txt for details.
*/

/***************************************************************************
 *   Jean Baptiste Filippi - 01.11.2005                                                           *
 *   Enrico Fucile
 *                                                                         *
 ***************************************************************************/
#include "grib_api_internal.h"
/*
   This is used by make_class.pl

   START_CLASS_DEF
   CLASS      = accessor
   IMPLEMENTS = unpack_long;pack_long; clear
   IMPLEMENTS = unpack_double;pack_double;unpack_double_element
   IMPLEMENTS = unpack_string;pack_string
   IMPLEMENTS = unpack_bytes;pack_bytes
   IMPLEMENTS = unpack_double_subarray 
   IMPLEMENTS = init;dump;destroy
   IMPLEMENTS = get_native_type;sub_section
   IMPLEMENTS = next_offset;value_count;byte_offset;byte_count
   IMPLEMENTS = notify_change;pack_expression
   IMPLEMENTS  = update_size; next; preferred_size
   IMPLEMENTS = compare;is_missing
   END_CLASS_DEF

 */

/* START_CLASS_IMP */

/*

Don't edit anything between START_CLASS_IMP and END_CLASS_IMP
Instead edit values between START_CLASS_DEF and END_CLASS_DEF
or edit "accessor.class" and rerun ./make_class.pl

*/

static grib_section* sub_section(grib_accessor* a);
static int  get_native_type(grib_accessor*);
static int is_missing(grib_accessor*);
static int pack_bytes(grib_accessor*,const unsigned char*, size_t *len);
static int pack_double(grib_accessor*, const double* val,size_t *len);
static int pack_long(grib_accessor*, const long* val,size_t *len);
static int pack_string(grib_accessor*, const char*, size_t *len);
static int pack_expression(grib_accessor*, grib_expression*);
static int unpack_bytes (grib_accessor*,unsigned char*, size_t *len);
static int unpack_double(grib_accessor*, double* val,size_t *len);
static int unpack_long(grib_accessor*, long* val,size_t *len);
static int unpack_string (grib_accessor*, char*, size_t *len);
static long byte_count(grib_accessor*);
static long byte_offset(grib_accessor*);
static long next_offset(grib_accessor*);
static long value_count(grib_accessor*);
static void destroy(grib_context*,grib_accessor*);
static void dump(grib_accessor*, grib_dumper*);
static void init(grib_accessor*,const long, grib_arguments* );
static void init_class(grib_accessor_class*);
static int notify_change(grib_accessor*,grib_accessor*);
static void update_size(grib_accessor*,size_t);
static size_t preferred_size(grib_accessor*,int);
static grib_accessor* next(grib_accessor*, int);
static int compare(grib_accessor*, grib_accessor*);
static int unpack_double_element(grib_accessor*,size_t i, double* val);
static int unpack_double_subarray(grib_accessor*, double* val,size_t start,size_t len);
static int clear(grib_accessor*);

typedef struct grib_accessor_gen {
    grib_accessor          att;
/* Members defined in gen */
} grib_accessor_gen;


static grib_accessor_class _grib_accessor_class_gen = {
    0,                      /* super                     */
    "gen",                      /* name                      */
    sizeof(grib_accessor_gen),  /* size                      */
    0,                           /* inited */
    &init_class,                 /* init_class */
    &init,                       /* init                      */
    0,                  /* post_init                      */
    &destroy,                    /* free mem                       */
    &dump,                       /* describes himself         */
    &next_offset,                /* get length of section     */
    &value_count,                /* get number of values      */
    &byte_count,                 /* get number of bytes      */
    &byte_offset,                /* get offset to bytes           */
    &get_native_type,            /* get native type               */
    &sub_section,                /* get sub_section                */
    0,               /* grib_pack procedures long      */
    &is_missing,               /* grib_pack procedures long      */
    &pack_long,                  /* grib_pack procedures long      */
    &unpack_long,                /* grib_unpack procedures long    */
    &pack_double,                /* grib_pack procedures double    */
    &unpack_double,              /* grib_unpack procedures double  */
    &pack_string,                /* grib_pack procedures string    */
    &unpack_string,              /* grib_unpack procedures string  */
    &pack_bytes,                 /* grib_pack procedures bytes     */
    &unpack_bytes,               /* grib_unpack procedures bytes   */
    &pack_expression,            /* pack_expression */
    &notify_change,              /* notify_change   */
    &update_size,                /* update_size   */
    &preferred_size,            /* preferred_size   */
    0,                    /* resize   */
    0,      /* nearest_smaller_value */
    &next,                       /* next accessor    */
    &compare,                    /* compare vs. another accessor   */
    &unpack_double_element,     /* unpack only ith value          */
    &unpack_double_subarray,     /* unpack a subarray         */
    &clear,             		/* clear          */
};


grib_accessor_class* grib_accessor_class_gen = &_grib_accessor_class_gen;


static void init_class(grib_accessor_class* c)
{
}

/* END_CLASS_IMP */

static void init(grib_accessor* a,const long len, grib_arguments* param)
{

 grib_action* act=(grib_action*)(a->creator);
 if (a->flags & GRIB_ACCESSOR_FLAG_TRANSIENT) {
    a->length = 0;
	if (!a->vvalue) 
		a->vvalue = grib_context_malloc_clear(a->parent->h->context,sizeof(grib_virtual_value));
    a->vvalue->type=grib_accessor_get_native_type(a);
    a->vvalue->length=len;
    if (act->default_value!=NULL) {
       const char* p = 0;
       size_t len = 1;
       long l;
       int ret=0;
       double d;
       char tmp[1024];
	   grib_expression* expression=grib_arguments_get_expression(a->parent->h,act->default_value,0);
       int type = grib_expression_native_type(a->parent->h,expression);
       switch(type) {
         case GRIB_TYPE_DOUBLE:
           grib_expression_evaluate_double(a->parent->h,expression,&d);
           grib_pack_double(a,&d,&len);
           break;

         case GRIB_TYPE_LONG:
           grib_expression_evaluate_long(a->parent->h,expression,&l);
           grib_pack_long(a,&l,&len);
           break;

         default:
           len = sizeof(tmp);
           p = grib_expression_evaluate_string(a->parent->h,expression,tmp,&len,&ret);
           if (ret != GRIB_SUCCESS) {
             grib_context_log(a->parent->h->context,GRIB_LOG_ERROR,"unable to evaluate %s as string",a->name);
             Assert(0);
           }
           len = strlen(p)+1;
           grib_pack_string(a,p,&len);
           break;
       }
    }
  } else
    a->length = len;
}

static void dump(grib_accessor* a, grib_dumper* dumper)
{
  if(a->cclass->unpack_string)
    grib_dump_string(dumper,a,NULL);
  else if(a->cclass->unpack_double)
    grib_dump_double(dumper,a,NULL);
  else if(a->cclass->unpack_long)
    grib_dump_long(dumper,a,NULL);
  else
    grib_dump_bytes(dumper,a,NULL);
}

static long next_offset(grib_accessor* a)
{
  return a->offset+a->length;
}

static long value_count(grib_accessor* a)
{
  return 1;
}

static long byte_count(grib_accessor* a)
{
  return a->length;
}

static int  get_native_type(grib_accessor* a){
  grib_context_log(a->parent->h->context,GRIB_LOG_ERROR,
      "Accessor %s [%s] must implement 'get_native_type'", a->name,a->cclass->name);
  return GRIB_TYPE_UNDEFINED;
}

static long byte_offset(grib_accessor* a)
{
  return a->offset;
}

static int unpack_bytes(grib_accessor* a, unsigned char* val, size_t *len)
{
  unsigned char* buf = a->parent->h->buffer->data;
  long length = grib_byte_count(a);
  long offset = grib_byte_offset(a);


  if(*len < length )
  {
    grib_context_log(a->parent->h->context, GRIB_LOG_ERROR, "Wrong size for %s it is %d bytes long\n", a->name ,length );
    *len = length;
    return GRIB_ARRAY_TOO_SMALL;
  }


  memcpy(val,buf + offset,length );
  *len = length;

  return GRIB_SUCCESS;
}

static int clear(grib_accessor* a)
{
  unsigned char* buf = a->parent->h->buffer->data;
  long length = grib_byte_count(a);
  long offset = grib_byte_offset(a);

  memset(buf + offset,0,length);

  return GRIB_SUCCESS;
}

static int  unpack_long   (grib_accessor* a, long*  v, size_t *len){

  if(a->cclass->unpack_double && a->cclass->unpack_double != &unpack_double)
  {
    double val = 0.0;
    size_t l = 1;
    grib_unpack_double (a , &val, &l);
    *v = (long)val;
    grib_context_log(a->parent->h->context,GRIB_LOG_DEBUG, " Casting double %s to long", a->name);
    return GRIB_SUCCESS;
  }

  if(a->cclass->unpack_string && a->cclass->unpack_string != &unpack_string)
  {
    char val[1024];
    size_t l = sizeof(val);
    char  *last = NULL;
    grib_unpack_string (a , val, &l);

    *v = strtol(val,&last,10);

    if(*last == 0)
    {
      grib_context_log(a->parent->h->context,GRIB_LOG_DEBUG, " Casting string %s to long", a->name);
      return GRIB_SUCCESS;
    }
  }

  return GRIB_NOT_IMPLEMENTED;
}

static int unpack_double (grib_accessor* a, double*v, size_t *len){

  if(a->cclass->unpack_long && a->cclass->unpack_long != &unpack_long)
  {
    long val = 0;
    size_t l = 1;
    grib_unpack_long (a , &val, &l);
    *v = val;
    grib_context_log(a->parent->h->context,GRIB_LOG_DEBUG, " Casting long %s to double", a->name);
    return GRIB_SUCCESS;
  }

  if(a->cclass->unpack_string && a->cclass->unpack_string != &unpack_string)
  {
    char val[1024];
    size_t l = sizeof(val);
    char  *last = NULL;
    grib_unpack_string (a , val, &l);

    *v = strtod(val,&last);

    if(*last == 0)
    {
      grib_context_log(a->parent->h->context,GRIB_LOG_DEBUG, " Casting string %s to long", a->name);
      return GRIB_SUCCESS;
    }
  }

  return GRIB_NOT_IMPLEMENTED;
}

static int unpack_string(grib_accessor*a , char*  v, size_t *len){

  if(a->cclass->unpack_double && a->cclass->unpack_double != &unpack_double)
  {
    double val = 0.0;
    size_t l = 1;
    grib_unpack_double (a , &val, &l);
    sprintf(v,"%g",val);
    *len = strlen(v);
    grib_context_log(a->parent->h->context,GRIB_LOG_DEBUG, " Casting double %s to string", a->name);
    return GRIB_SUCCESS;
  }

  if(a->cclass->unpack_long && a->cclass->unpack_long != &unpack_long)
  {
    long val = 0;
    size_t l = 1;
    grib_unpack_long (a , &val, &l);
    sprintf(v,"%ld",val);
    *len = strlen(v);
    grib_context_log(a->parent->h->context,GRIB_LOG_DEBUG, " Casting long %s to string  \n", a->name);
    return GRIB_SUCCESS;
  }

  return GRIB_NOT_IMPLEMENTED;
}

static int pack_expression(grib_accessor* a, grib_expression *e){
  size_t len = 1;
  long   lval;
  double   dval;
  const char    *cval;
  int ret=0;
  char tmp[1024];

  switch(grib_accessor_get_native_type(a))
  {
    case GRIB_TYPE_LONG:
      len = 1;
      ret = grib_expression_evaluate_long(a->parent->h,e,&lval);
      if (ret != GRIB_SUCCESS) {
        grib_context_log(a->parent->h->context,GRIB_LOG_ERROR,"unable to set %s as long",a->name);
        return ret;
      }
      return grib_pack_long(a,&lval,&len);
      break;

    case GRIB_TYPE_DOUBLE:
      len = 1;
      ret = grib_expression_evaluate_double(a->parent->h,e,&dval);
      return grib_pack_double(a,&dval,&len);
      break;

    case GRIB_TYPE_STRING:
      len = sizeof(tmp);
      cval = grib_expression_evaluate_string(a->parent->h,e,tmp,&len,&ret);
      if (ret != GRIB_SUCCESS) {
        grib_context_log(a->parent->h->context,GRIB_LOG_ERROR,"unable to set %s as string",a->name);
        return ret;
      }
      len = strlen(cval);
      return grib_pack_string(a,cval,&len);
      break;
  }

  return GRIB_NOT_IMPLEMENTED;
}

static int pack_long(grib_accessor* a, const long*  v, size_t *len){
  grib_context* c=a->parent->h->context;
  if(a->cclass->pack_double && a->cclass->pack_double != &pack_double)
  {
    int i=0,ret=0;
    double* val = (double*)grib_context_malloc(c,*len*(sizeof(double))) ;
    if (!val) {
      grib_context_log(c,GRIB_LOG_ERROR,
       "unable to allocate %d bytes\n",(int)(*len*(sizeof(double))));
      return GRIB_OUT_OF_MEMORY;
    }
    for (i=0;i<*len;i++) val[i]=(long)v[i];
    ret=grib_pack_double (a , val, len);
    grib_context_free(c,val);
    return ret;
  }
  grib_context_log(c,GRIB_LOG_ERROR, " Should not grib_pack %s as long", a->name);
  Assert(0);
  return GRIB_NOT_IMPLEMENTED;
}

static int pack_double(grib_accessor* a, const double *v, size_t *len){
  grib_context* c=a->parent->h->context;
  if(a->cclass->pack_long && a->cclass->pack_long != &pack_long)
  {
    int i=0,ret=0;
    long* val = (long*)grib_context_malloc(c,*len*(sizeof(long))) ;
    if (!val) {
      grib_context_log(c,GRIB_LOG_ERROR,
       "unable to allocate %d bytes\n",(int)(*len*(sizeof(long))));
      return GRIB_OUT_OF_MEMORY;
    }
    for (i=0;i<*len;i++) val[i]=(long)v[i];
    ret=grib_pack_long (a , val, len);
    grib_context_free(c,val);
    return ret;
  }
  grib_context_log(c,GRIB_LOG_ERROR, " Should not grib_pack %s  as double", a->name);
  return GRIB_NOT_IMPLEMENTED;
}

static int pack_string(grib_accessor*a , const char*  v, size_t *len){
  if(a->cclass->pack_double && a->cclass->pack_double != &pack_double)
  {
    size_t l = 1;
    double val = atof(v);
    return grib_pack_double (a , &val, &l);
  }

  if(a->cclass->pack_long && a->cclass->pack_long != &pack_long)
  {
    size_t l = 1;
    long val = atof(v);
    return grib_pack_long (a , &val, &l);
  }

  grib_context_log(a->parent->h->context,GRIB_LOG_ERROR,
      " Should not grib_pack %s  as string", a->name);
  return GRIB_NOT_IMPLEMENTED;
}

static int pack_bytes(grib_accessor* a, const unsigned char* val, size_t *len)
{
  size_t length = *len;
  grib_buffer_replace(a, val, length,1,1);
  return GRIB_SUCCESS;
}

static void destroy(grib_context* ct, grib_accessor* a)
{
  grib_dependency_remove_observed(a);
  grib_dependency_remove_observer(a);
  if (a->vvalue!=NULL) {
  	grib_context_free(ct,a->vvalue);
	a->vvalue=NULL;
  }
}

static grib_section* sub_section(grib_accessor* a)
{
  return NULL;
}

static int notify_change(grib_accessor* self,grib_accessor* observed)
{
  /* Default behaviour is to notify creator */
  return grib_action_notify_change(self->creator,self,observed);
}

static void update_size(grib_accessor* a,size_t s)
{
  grib_context_log(a->parent->h->context,GRIB_LOG_ERROR,
      "Accessor %s [%s] must implement 'update_size'", a->name,a->cclass->name);
  Assert(0 == 1);
}

static grib_accessor* next(grib_accessor* a, int mod) {
   grib_accessor* next=NULL;
   if (a->next) {
     next=a->next;
   } else {
     if (a->parent->owner)
      next=a->parent->owner->cclass->next(a->parent->owner,0);
   }
   return next;
}


static int compare(grib_accessor* a, grib_accessor* b) {
  return GRIB_NOT_IMPLEMENTED;
}

/* Redefined in all padding */

static size_t preferred_size(grib_accessor* a,int from_handle)
{
  return a->length;
}

static int is_missing(grib_accessor* a){
    int i=0;
    int is_missing=1;
    unsigned char ones=0xff;
    unsigned char* v=NULL;

    if (a->flags & GRIB_ACCESSOR_FLAG_TRANSIENT) {
	  if (a->vvalue == NULL) {
	    grib_context_log(a->parent->h->context,GRIB_LOG_ERROR,"%s internal error (flags=0x%X)",a->name,a->flags);
	  }
	  Assert(a->vvalue!=NULL);
	  return a->vvalue->missing;
	}
    Assert(a->length>=0);

    v=a->parent->h->buffer->data+a->offset;

    for (i=0; i < a->length; i++) {
      if (*v != ones) {
        is_missing=0;
        break;
      }
      v++;
    }

    return is_missing;
}

static int unpack_double_element(grib_accessor* a, size_t i, double* val) {
  return GRIB_NOT_IMPLEMENTED;
}

static int unpack_double_subarray(grib_accessor* a, double* val,size_t start,size_t len) {
  return GRIB_NOT_IMPLEMENTED;
}