#include "../../include/sane/config.h"

#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <string.h>
#include <assert.h>

/* sane includes for the sanei functions called */
#include "../include/sane/sane.h"
#include "../include/sane/saneopts.h"
#include "../include/sane/sanei.h"
#include "../include/sane/sanei_config.h"

#define XSTR(s) STR(s)
#define STR(s) #s
#define CONFIG_PATH XSTR(TESTSUITE_SANEI_SRCDIR)

/*
 * variables and functions used by the tests below
 */


/* range for constraint */
static const SANE_Range model_range = {
  1000,				/* minimum */
  2000,				/* maximum */
  2				/* quantization */
};

/* range for memory buffer size constraint */
static const SANE_Range buffer_range = {
  1024,				/* minimum bytes */
  2048 * 1024,			/* maximum bytes */
  1024				/* quantization */
};

/* range for int value in [0-15] */
static const SANE_Range value16_range = {
  0,				/* minimum */
  15,				/* maximum */
  1				/* quantization */
};

/* range for fixed height value */
static const SANE_Range height_range = {
  SANE_FIX (0),			/* minimum */
  SANE_FIX (29.7),		/* maximum */
  0				/* no quantization : hard to do for float values ... */
};

/* list of astra models */
static const SANE_String_Const astra_models[] =
  { "610", "1220", "1600", "2000", NULL };

/* string list */
static const SANE_String_Const string_list[] =
  { "string1", "string2", "string3", "string4", NULL };

/* last device name used for attach callback */
static char *lastdevname = NULL;

static SANE_Status
check_config_attach (SANEI_Config * config, const char *devname)
{
  /* silence compiler warning for now */
  if (config == NULL)
    {
      return SANE_STATUS_INVAL;
    }

  fprintf (stdout, "attaching with devname '%s'\n", devname);
  if (lastdevname != NULL)
    {
      free (lastdevname);
    }
  lastdevname = strdup (devname);
  return SANE_STATUS_GOOD;
}

/******************************/
/* start of tests definitions */
/******************************/

/*
 * non-existent config file
 */
static void
inexistent_config (void)
{
  SANE_Status status;
  SANEI_Config config;

  config.count = 0;
  config.descriptors = NULL;
  config.values = NULL;
  status = sanei_configure_attach (CONFIG_PATH
                                   "/data/inexistent.conf", &config, NULL);

  /* check results */
  assert (status != SANE_STATUS_GOOD);
}


/*
 * no config struct
 */
static void
null_config (void)
{
  SANE_Status status;

  status =
    sanei_configure_attach (CONFIG_PATH "/data/umax_pp.conf", NULL,
                            check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
}


/*
 * no attach function
 */
static void
null_attach (void)
{
  SANE_Status status;

  status = sanei_configure_attach (CONFIG_PATH
                                   "/data/umax_pp.conf", NULL, NULL);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
}


/*
 * empty config : backend has no configuration option
 */
static void
empty_config (void)
{
  SANE_Status status;
  SANEI_Config config;

  config.count = 0;
  config.descriptors = NULL;
  config.values = NULL;
  status =
    sanei_configure_attach (CONFIG_PATH "/data/empty.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
}


/*
 * string option
 */
static void
string_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Char modelname[128];
  SANE_Char vendor[128];
  SANE_Option_Descriptor *options[2];
  void *values[2];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "modelname";
  options[i]->title = "model name";
  options[i]->desc = "user provided scanner's model name";
  options[i]->type = SANE_TYPE_STRING;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = 128;
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = modelname;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "vendor";
  options[i]->title = "vendor name";
  options[i]->desc = "user provided scanner's vendor name";
  options[i]->type = SANE_TYPE_STRING;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = 128;
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = vendor;
  i++;

  config.count = i;
  config.descriptors = options;
  config.values = values;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/string.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
  assert (strcmp (modelname, "my model") == 0);
  assert (strcmp (vendor, "my vendor") == 0);
}


/*
 * int option
 */
static void
int_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Word modelnumber;
  SANE_Option_Descriptor *options[1];
  void *values[1];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "modelnumber";
  options[i]->title = "model number";
  options[i]->desc = "user provided scanner's model number";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &model_range;
  values[i] = &modelnumber;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/int.conf", &config,
                            check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
  assert (modelnumber == 1234);
}


/*
 * int option out of range
 */
static void
wrong_range_int_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Word modelnumber = -1;
  SANE_Option_Descriptor *options[1];
  void *values[1];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "modelnumber";
  options[i]->title = "model number";
  options[i]->desc = "user provided scanner's model number";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &model_range;
  values[i] = &modelnumber;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/wrong-range.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_INVAL);
  assert (modelnumber == -1);
}


/*
 * word array
 */
static void
word_array_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Word numbers[7];
  SANE_Option_Descriptor *options[1];
  void *values[1];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "numbers";
  options[i]->title = "some numbers";
  options[i]->desc = "an array of numbers";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word) * 7;
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &model_range;
  values[i] = numbers;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/word-array.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
  for (i = 0; i < 7; i++)
    {
      assert (numbers[i] == 1000 + 100 * i);
    }
}


/*
 * string option with string list constraint
 */
static void
string_list_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Char choice[128];
  SANE_Option_Descriptor *options[1];
  void *values[1];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "string-choice";
  options[i]->title = "string choice";
  options[i]->desc = "one string among a fixed list";
  options[i]->type = SANE_TYPE_STRING;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = 128;
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_STRING_LIST;
  options[i]->constraint.string_list = string_list;
  values[i] = choice;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/string-list.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
  assert (strcmp (choice, "string3") == 0);
}


/*
 * string option with string list constraint
 */
static void
wrong_string_list_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Char choice[128];
  SANE_Option_Descriptor *options[1];
  void *values[1];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "string-choice";
  options[i]->title = "string choice";
  options[i]->desc = "one string among a fixed list";
  options[i]->type = SANE_TYPE_STRING;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = 128;
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_STRING_LIST;
  options[i]->constraint.string_list = string_list;
  values[i] = choice;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  choice[0] = 0;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH
                            "/data/wrong-string-list.conf", &config,
			    check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_INVAL);
  assert (strcmp (choice, "") == 0);
}


/*
 * real umax_pp confiugration file parsing
 */
static void
umax_pp (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Option_Descriptor *options[9];
  void *values[9];
  int i = 0;
  /* placeholders for options */
  SANE_Word buffersize = -1;
  SANE_Word redgain = -1;
  SANE_Word greengain = -1;
  SANE_Word bluegain = -1;
  SANE_Word redoffset = -1;
  SANE_Word greenoffset = -1;
  SANE_Word blueoffset = -1;
  SANE_Char model[128];

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "buffer";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &buffer_range;
  values[i] = &buffersize;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "red-gain";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &value16_range;
  values[i] = &redgain;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "green-gain";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &value16_range;
  values[i] = &greengain;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "blue-gain";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &value16_range;
  values[i] = &bluegain;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "red-offset";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &value16_range;
  values[i] = &redoffset;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "green-offset";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &value16_range;
  values[i] = &greenoffset;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "blue-offset";
  options[i]->type = SANE_TYPE_INT;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &value16_range;
  values[i] = &blueoffset;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "astra";
  options[i]->type = SANE_TYPE_STRING;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = 128;
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_STRING_LIST;
  options[i]->constraint.string_list = astra_models;
  values[i] = &model;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  model[0] = 0;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/umax_pp.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
  assert (buffersize == 1048576);
  assert (redgain == 1);
  assert (greengain == 2);
  assert (bluegain == 3);
  assert (redoffset == 4);
  assert (greenoffset == 5);
  assert (blueoffset == 6);
  assert (strcmp (model, "1600") == 0);
  assert (strcmp (lastdevname, "port safe-auto") == 0);

  /* free memory */
  while (i > 0)
    {
      i--;
      free (options[i]);
    }
}


/*
 * boolean option
 */
static void
wrong_bool_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Option_Descriptor *options[2];
  void *values[2];
  SANE_Bool booltrue, boolfalse;
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "booltrue";
  options[i]->title = "boolean true";
  options[i]->type = SANE_TYPE_BOOL;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Bool);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = &booltrue;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "boolfalse";
  options[i]->title = "boolean false";
  options[i]->type = SANE_TYPE_BOOL;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Bool);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = &boolfalse;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/wrong-boolean.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_INVAL);
  assert (booltrue == SANE_TRUE);
}


/*
 * boolean option
 */
static void
bool_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Option_Descriptor *options[3];
  void *values[3];
  SANE_Bool booltrue, boolfalse, boolarray[3];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "booltrue";
  options[i]->title = "boolean true";
  options[i]->type = SANE_TYPE_BOOL;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Bool);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = &booltrue;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "boolfalse";
  options[i]->title = "boolean false";
  options[i]->type = SANE_TYPE_BOOL;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Bool);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = &boolfalse;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "boolarray";
  options[i]->title = "boolean array";
  options[i]->type = SANE_TYPE_BOOL;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (boolarray);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = boolarray;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/boolean.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
  assert (booltrue == SANE_TRUE);
  assert (boolfalse == SANE_FALSE);
  for (i = 0; i < 3; i++)
    {
      assert (boolarray[i] == (SANE_Bool) i % 2);
    }
}


/*
 * fixed option
 */
static void
fixed_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Word width, height, fixedarray[7];
  SANE_Option_Descriptor *options[3];
  void *values[3];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "width";
  options[i]->title = "width";
  options[i]->type = SANE_TYPE_FIXED;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = &width;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "height";
  options[i]->title = "height";
  options[i]->type = SANE_TYPE_FIXED;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = &height;
  i++;

  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "array-of-fixed";
  options[i]->title = "array of fixed";
  options[i]->type = SANE_TYPE_FIXED;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (fixedarray);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &height_range;
  values[i] = fixedarray;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/fixed.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
  assert (width == SANE_FIX (21.0));
  assert (height == SANE_FIX (29.7));
  for (i = 0; i < 7; i++)
    {
      assert (fixedarray[i] == SANE_FIX (2.0 + 0.1 * ((float) i)));
    }
}


/*
 * fixed option with value out of range
 */
static void
wrong_fixed_option (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Word height;
  SANE_Option_Descriptor *options[1];
  void *values[1];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "height";
  options[i]->title = "height";
  options[i]->type = SANE_TYPE_FIXED;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (SANE_Word);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_RANGE;
  options[i]->constraint.range = &height_range;
  values[i] = &height;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/wrong-fixed.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_INVAL);
}


static void
snapscan (void)
{
  SANE_Status status;
  SANEI_Config config;
  SANE_Char firmware[128];
  SANE_Option_Descriptor *options[1];
  void *values[1];
  int i;

  i = 0;
  options[i] =
    (SANE_Option_Descriptor *) malloc (sizeof (SANE_Option_Descriptor));
  options[i]->name = "firmware";
  options[i]->title = "scanner's firmware path";
  options[i]->desc = "user provided scanner's full path";
  options[i]->type = SANE_TYPE_STRING;
  options[i]->unit = SANE_UNIT_NONE;
  options[i]->size = sizeof (firmware);
  options[i]->cap = SANE_CAP_SOFT_SELECT;
  options[i]->constraint_type = SANE_CONSTRAINT_NONE;
  values[i] = firmware;
  i++;

  config.descriptors = options;
  config.values = values;
  config.count = i;

  /* configure and attach */
  status =
    sanei_configure_attach (CONFIG_PATH "/data/snapscan.conf",
                            &config, check_config_attach);

  /* check results */
  assert (status == SANE_STATUS_GOOD);
  assert (strcmp (firmware, "/usr/share/sane/snapscan/your-firmwarefile.bin")
	  == 0);
  /* TODO must test attach() done */
}


/**
 * create the test suite for sanei config related tests 
 */
static void
sanei_config_suite (void)
{
  /* tests */
  inexistent_config ();
  empty_config ();
  null_config ();
  null_attach ();
  string_option ();
  int_option ();
  string_list_option ();
  word_array_option ();
  bool_option ();
  fixed_option ();
  wrong_range_int_option ();
  wrong_string_list_option ();
  wrong_bool_option ();
  wrong_fixed_option ();

  /* backend real conf inspired cases */
  umax_pp ();
  snapscan ();
}

/**
 * main function to run the test suites 
 */
int
main (void)
{
  /* run suites */
  sanei_config_suite ();

  return 0;
}

/* vim: set sw=2 cino=>2se-1sn-1s{s^-1st0(0u0 smarttab expandtab: */