/*max@home*/
#ifdef __cplusplus
extern "C" {
#endif
#ifdef NOTE
ConvertOneValue2Enum() potrebbe essere fonte di errore se
DCItemSize[one_value.ItemType] >
DCItemSize[TW_INT8] e si utilizzano i bit piu significativi di one_value
.Item
ConvertEnumeration2Range() potrebbe restituire valori di StepSize non
appropriati se enumeration.ItemType e TW_FIX32 potrebbe
restituire valori MinValue non corretti se DCItemSize[xxx.ItemType] >
DCItemSize[TW_INT8] e si utilizzano i bit piu significativi di xxx.Item
ConvertEnum2OneValue ritorna(ovviamente) solo il valore corrente
#endif
#include <assert.h>
#include <string.h>
#include "ttwain_capability.h"
#include "ttwain_conversion.h"
#include "ttwain_error.h"
#include "ttwain_state.h"
#include "ttwain_global_def.h"
/*---------------------------------------------------------------------------*/
static const size_t DCItemSize[13] = {
sizeof(TW_INT8), sizeof(TW_INT16), sizeof(TW_INT32), sizeof(TW_UINT8),
sizeof(TW_UINT16), sizeof(TW_UINT32), sizeof(TW_BOOL), sizeof(TW_FIX32),
sizeof(TW_FRAME), sizeof(TW_STR32), sizeof(TW_STR64), sizeof(TW_STR128),
sizeof(TW_STR255),
}; /* see twain.h */
/*---------------------------------------------------------------------------*/
#define TWON_TWON(TYPE1, TYPE2) (((TYPE1) << 8) | (TYPE2))
/*---------------------------------------------------------------------------*/
static int ConvertOneValue2Range(TW_ONEVALUE one_value, TW_RANGE *range);
static int ConvertEnumeration2Range(TW_ENUMERATION enumeration,
TW_RANGE *range);
static int ConvertOneValue2Enum(TW_ONEVALUE one_value, TW_ENUMERATION *tw_enum);
static int ConvertEnum2OneValue(TW_ENUMERATION tw_enum, TW_ONEVALUE *one_value);
static int ConvertEnum2Array(TW_ENUMERATION tw_enum, TW_ARRAY *array);
static TUINT32 GetContainerSize(int nFormat, unsigned twty, TW_UINT32 nItems);
static int TTWAIN_GetCapability(TW_INT16 msgType, TW_UINT16 cap_id,
TW_UINT16 conType, void *data,
TUINT32 *cont_size);
/*------------------------------------------------------------------------*/
int TTWAIN_GetCap(TW_UINT16 cap_id, TW_UINT16 conType, void *data,
TUINT32 *cont_size) {
return TTWAIN_GetCapability(MSG_GET, cap_id, conType, data, cont_size);
}
/*------------------------------------------------------------------------*/
int TTWAIN_GetCapCurrent(TW_UINT16 cap_id, TW_UINT16 conType, void *data,
TUINT32 *cont_size) {
return TTWAIN_GetCapability(MSG_GETCURRENT, cap_id, conType, data, cont_size);
}
/*------------------------------------------------------------------------*/
int TTWAIN_GetCapQuery(TW_UINT16 cap_id, TW_UINT16 *pattern) {
int rc;
/* GCC9 during compilation shows that this code possible call
* TTWAIN_GetCapability() with case TWON_TWON(TWON_RANGE, TWON_RANGE)
* whitch cause stack corruption, so make 'data' big enough to store
* TW_ONEVALUE. */
TW_ONEVALUE data[1 + (sizeof(TW_RANGE) / sizeof(TW_ONEVALUE))];
rc = TTWAIN_GetCapability(MSG_QUERYSUPPORT, cap_id, TWON_ONEVALUE, &data, 0);
if (!rc) return FALSE;
*pattern = (TW_UINT16)data[0].Item;
return TRUE;
}
/*------------------------------------------------------------------------*/
static int TTWAIN_GetCapability(TW_INT16 msgType, TW_UINT16 cap_id,
TW_UINT16 conType, void *data,
TUINT32 *cont_size) {
TW_CAPABILITY cap;
void *pv;
TW_ENUMERATION *my_enum;
TW_ARRAY *my_array;
TW_ONEVALUE *my_one;
TW_RANGE *my_range;
TUINT32 size = 0;
if (!data && !cont_size) return FALSE;
if (TTWAIN_GetState() < TWAIN_SOURCE_OPEN) {
TTWAIN_ErrorBox("Attempt to get capability value below State 4.");
return FALSE;
}
/* Fill in capability structure */
cap.Cap = cap_id; /* capability id */
cap.ConType =
TWON_DONTCARE16; /* favorite type of container (should be ignored...) */
cap.hContainer = NULL;
if (TTWAIN_DS(DG_CONTROL, DAT_CAPABILITY, msgType, (TW_MEMREF)&cap) !=
TWRC_SUCCESS)
return FALSE;
if (!cap.hContainer) return FALSE;
if (msgType == MSG_QUERYSUPPORT) {
}
pv = GLOBAL_LOCK(cap.hContainer);
my_enum = (TW_ENUMERATION *)pv;
my_array = (TW_ARRAY *)pv;
my_one = (TW_ONEVALUE *)pv;
my_range = (TW_RANGE *)pv;
if (cont_size) {
switch (TWON_TWON(cap.ConType, conType)) {
case TWON_TWON(TWON_ENUMERATION, TWON_ENUMERATION):
*cont_size = GetContainerSize(TWON_ENUMERATION, my_enum->ItemType,
my_enum->NumItems);
break;
case TWON_TWON(TWON_ONEVALUE, TWON_ENUMERATION):
*cont_size = GetContainerSize(TWON_ENUMERATION, my_one->ItemType, 1);
break;
case TWON_TWON(TWON_ARRAY, TWON_ARRAY):
*cont_size =
GetContainerSize(TWON_ARRAY, my_array->ItemType, my_array->NumItems);
break;
case TWON_TWON(TWON_ONEVALUE, TWON_ONEVALUE):
*cont_size = GetContainerSize(TWON_ONEVALUE, my_one->ItemType, 1);
break;
case TWON_TWON(TWON_ENUMERATION, TWON_ARRAY):
*cont_size =
GetContainerSize(TWON_ARRAY, my_enum->ItemType, my_enum->NumItems);
break;
default:
/* tmsg_error("Unable to convert type %d to %d (cap 0x%x)\n",
* cap.ConType, conType,cap_id);*/
assert(0);
GLOBAL_UNLOCK(cap.hContainer);
GLOBAL_FREE(cap.hContainer);
return FALSE;
}
GLOBAL_UNLOCK(cap.hContainer);
GLOBAL_FREE(cap.hContainer);
return TRUE;
}
switch (TWON_TWON(cap.ConType, conType)) {
case TWON_TWON(TWON_ENUMERATION, TWON_ENUMERATION):
size = GetContainerSize(cap.ConType, my_enum->ItemType, my_enum->NumItems);
memcpy(data, my_enum, size);
break;
case TWON_TWON(TWON_ENUMERATION, TWON_RANGE):
ConvertEnumeration2Range(*my_enum, (TW_RANGE *)data);
break;
case TWON_TWON(TWON_ENUMERATION, TWON_ONEVALUE):
ConvertEnum2OneValue(*my_enum, (TW_ONEVALUE *)data);
break;
case TWON_TWON(TWON_ARRAY, TWON_ARRAY):
size =
GetContainerSize(cap.ConType, my_array->ItemType, my_array->NumItems);
memcpy(data, my_array, size);
break;
case TWON_TWON(TWON_ONEVALUE, TWON_ONEVALUE):
memcpy(data, my_one, sizeof(TW_ONEVALUE));
break;
case TWON_TWON(TWON_ONEVALUE, TWON_RANGE):
ConvertOneValue2Range(*my_one, (TW_RANGE *)data);
break;
case TWON_TWON(TWON_ONEVALUE, TWON_ENUMERATION):
ConvertOneValue2Enum(*my_one, (TW_ENUMERATION *)data);
break;
case TWON_TWON(TWON_RANGE, TWON_RANGE):
memcpy(data, my_range, sizeof(TW_RANGE));
break;
case TWON_TWON(TWON_ENUMERATION, TWON_ARRAY):
ConvertEnum2Array(*my_enum, (TW_ARRAY *)data);
break;
default:
assert(0);
GLOBAL_UNLOCK(cap.hContainer);
GLOBAL_FREE(cap.hContainer);
return FALSE;
}
GLOBAL_UNLOCK(cap.hContainer);
GLOBAL_FREE(cap.hContainer);
return TRUE;
}
/*---------------------------------------------------------------------------*/
/*---------------------------------------------------------------------------*/
/*---------------------------------------------------------------------------*/
/*---------------------------------------------------------------------------*/
static int ConvertOneValue2Range(TW_ONEVALUE one_value, TW_RANGE *range) {
range->ItemType = one_value.ItemType;
range->MinValue = one_value.Item;
range->MaxValue = one_value.Item;
range->StepSize = 0;
range->DefaultValue = one_value.Item;
range->CurrentValue = one_value.Item;
return TRUE;
}
/*---------------------------------------------------------------------------*/
static int ConvertEnumeration2Range(TW_ENUMERATION enumeration,
TW_RANGE *range) {
range->ItemType = enumeration.ItemType;
range->MinValue = enumeration.ItemList[0];
range->MaxValue = enumeration.ItemList[enumeration.NumItems - 1];
range->StepSize = (range->MaxValue - range->MinValue) / enumeration.NumItems;
if (range->MaxValue < range->MinValue) {
range->MaxValue = range->MinValue;
range->StepSize = 0;
}
range->DefaultValue = enumeration.ItemList[enumeration.DefaultIndex];
range->CurrentValue = enumeration.ItemList[enumeration.CurrentIndex];
return TRUE;
}
/*---------------------------------------------------------------------------*/
static int ConvertOneValue2Enum(TW_ONEVALUE one_value,
TW_ENUMERATION *tw_enum) {
tw_enum->ItemType = one_value.ItemType;
tw_enum->NumItems = 1;
tw_enum->CurrentIndex = 0; /* Current value is in ItemList[CurrentIndex] */
tw_enum->DefaultIndex = 0; /* Powerup value is in ItemList[DefaultIndex] */
tw_enum->ItemList[0] = (TW_UINT8)one_value.Item;
return TRUE;
}
/*---------------------------------------------------------------------------*/
static int ConvertEnum2OneValue(TW_ENUMERATION tw_enum,
TW_ONEVALUE *one_value) {
unsigned char *base;
size_t ofs;
size_t itemSize;
itemSize = DCItemSize[tw_enum.ItemType];
base = tw_enum.ItemList;
ofs = tw_enum.CurrentIndex * itemSize;
one_value->ItemType = tw_enum.ItemType;
one_value->Item = 0;
memcpy(&(one_value->Item), &(base[ofs]), itemSize);
return TRUE;
}
/*---------------------------------------------------------------------------*/
static int ConvertEnum2Array(TW_ENUMERATION tw_enum, TW_ARRAY *array) {
size_t itemSize;
size_t listSize;
itemSize = DCItemSize[tw_enum.ItemType];
listSize = itemSize * tw_enum.NumItems;
array->ItemType = tw_enum.ItemType;
array->NumItems = tw_enum.NumItems;
memcpy(&(array->ItemList), &(tw_enum.ItemList), listSize);
return TRUE;
}
/*---------------------------------------------------------------------------*/
/*---------------------------------------------------------------------------*/
/*---------------------------------------------------------------------------*/
/* SET CAP */
/*---------------------------------------------------------------------------*/
int TTWAIN_SetCap(TW_UINT16 cap_id, TW_UINT16 conType, TW_UINT16 itemType,
TW_UINT32 *value) {
int rc = FALSE;
TUINT32 size;
TW_CAPABILITY *capability = 0;
TW_HANDLE capabilityH = 0;
TW_ONEVALUE *container = 0;
TW_HANDLE containerH = 0;
size = GetContainerSize(conType, itemType, 1);
containerH = GLOBAL_ALLOC(GMEM_FIXED, size);
if (!containerH) goto done;
container = (TW_ONEVALUE *)GLOBAL_LOCK(containerH);
if (!container) goto done;
container->ItemType = itemType;
container->Item = *value;
capabilityH = GLOBAL_ALLOC(GMEM_FIXED, sizeof(TW_CAPABILITY));
if (!capabilityH) {
GLOBAL_UNLOCK(containerH);
GLOBAL_FREE(containerH);
containerH = NULL;
goto done;
}
capability = (TW_CAPABILITY *)GLOBAL_LOCK(capabilityH);
if (!capability) goto done;
capability->ConType = conType;
capability->hContainer = containerH;
if (TTWAIN_GetState() < TWAIN_SOURCE_OPEN) {
/*TTWAIN_ErrorBox("Setting capability in State < 4.");*/
TTWAIN_OpenSourceManager(0); /* Bring up to state 4 */
/*goto done;*/
}
capability->Cap = cap_id; /* capability id */
rc = (TTWAIN_DS(DG_CONTROL, DAT_CAPABILITY, MSG_SET, (TW_MEMREF)capability) ==
TWRC_SUCCESS);
done:
if (containerH) {
GLOBAL_UNLOCK(containerH);
GLOBAL_FREE(containerH);
}
if (capabilityH) {
GLOBAL_UNLOCK(capabilityH);
GLOBAL_FREE(capabilityH);
}
return rc;
}
/*---------------------------------------------------------------------------*/
static TUINT32 GetContainerSize(int nFormat, unsigned twty, TW_UINT32 nItems) {
size_t size;
switch (nFormat) {
case TWON_ONEVALUE:
size = sizeof(TW_ONEVALUE);
if (DCItemSize[twty] > sizeof(TW_UINT32)) {
size += DCItemSize[twty] - sizeof(TW_UINT32);
}
break;
case TWON_RANGE:
size = sizeof(TW_RANGE);
break;
case TWON_ENUMERATION:
size =
sizeof(TW_ENUMERATION) + DCItemSize[twty] * nItems - sizeof(TW_UINT8);
break;
case TWON_ARRAY:
size = sizeof(TW_ARRAY) + DCItemSize[twty] * nItems - sizeof(TW_UINT8);
break;
default:
size = 0;
break;
} /* switch */
return (TUINT32)size;
}
/*---------------------------------------------------------------------------*/
#ifdef __cplusplus
}
#endif