First attempt at reading TW_INFO, probably not correct.

This commit is contained in:
Eugene Wang 2023-04-09 17:38:43 -04:00
parent a72e1ff1d0
commit b8af6269f8
3 changed files with 96 additions and 23 deletions

View File

@ -1509,6 +1509,10 @@ namespace NTwain.Data
private TW_INFO Info_097; private TW_INFO Info_097;
private TW_INFO Info_098; private TW_INFO Info_098;
private TW_INFO Info_099; private TW_INFO Info_099;
// 200 seems overkill for anyone to request
// at once so I cut it down in half
//private TW_INFO Info_100; //private TW_INFO Info_100;
//private TW_INFO Info_101; //private TW_INFO Info_101;
//private TW_INFO Info_102; //private TW_INFO Info_102;

View File

@ -881,6 +881,7 @@ namespace NTwain.Data
/// <param name="memMgr"></param> /// <param name="memMgr"></param>
public void Free(IMemoryManager memMgr) public void Free(IMemoryManager memMgr)
{ {
#region don't open this
Info_000.Free(memMgr); Info_000.Free(memMgr);
Info_001.Free(memMgr); Info_001.Free(memMgr);
Info_002.Free(memMgr); Info_002.Free(memMgr);
@ -981,42 +982,110 @@ namespace NTwain.Data
Info_097.Free(memMgr); Info_097.Free(memMgr);
Info_098.Free(memMgr); Info_098.Free(memMgr);
Info_099.Free(memMgr); Info_099.Free(memMgr);
#endregion
} }
} }
partial struct TW_INFO partial struct TW_INFO
{ {
//public unsafe TValue ReadTWTYData<TValue>(int itemIndex = 0) where TValue : struct /// <summary>
//{ /// Quick check to see if the <see cref="Item"/> pointer is really
// if (ReturnCode != TWRC.SUCCESS || NumItems == 0) return default; /// a pointer or actual data (ugh).
/// </summary>
// var sz = ItemType.GetItemTypeSize(); public bool IsDataAPointer =>
// if (ItemType == TWTY.HANDLE || (sz * NumItems) > 4) ItemType == TWTY.HANDLE || (ItemType.GetItemTypeSize() * NumItems) > IntPtr.Size; // should it be intptr.size or just 4?
// {
// // read as ptr
// // this is probably not correct
// //((IntPtr)Item.ToPointer()).ReadTWTYData<TValue>(ItemType;
// }
// else
// {
// // read as 4 byte number
// }
//}
/// <summary> /// <summary>
/// Frees any memory used by this if necessary. /// Try to read out the item as the type specified in <see cref="ItemType"/>.
/// This ONLY works if the data is not a pointer (see <see cref="IsDataAPointer"/>).
/// For pointers you'd read it yourself with
/// <see cref="ValueReader.ReadTWTYData{TValue}(IntPtr, TWTY, int)"/>.
/// Unless it's a handle (<see cref="TWTY.HANDLE"/>) to non-twain-strings, then you'd use
/// <see cref="ReadHandleString(int)"/>.
/// </summary>
/// <param name="type"></param>
/// <typeparam name="TValue"></typeparam>
/// <returns></returns>
public unsafe TValue ReadNonPointerData<TValue>() where TValue : struct
{
if (ReturnCode != TWRC.SUCCESS || NumItems == 0 || IsDataAPointer) return default;
// we can try a trick and make a pointer to this numeric data
// and re-use our pointer reader. There's a good chance this is wrong in many ways.
// TODO: test this idea in some unit test
var value = TWPlatform.Is32bit ? Item.ToUInt32() : Item.ToUInt64(); // the value should be 32bit from the spec but not sure how it'll work in 64bit
var fakePtr = (IntPtr)(&value);
return fakePtr.ReadTWTYData<TValue>(ItemType, 0);
}
/// <summary>
/// Try to read a null-terminated string from the item.
/// </summary>
/// <param name="memMgr"></param>
/// <param name="index">If item is an array specify which string to read</param>
/// <returns></returns>
public unsafe string? ReadHandleString(IMemoryManager memMgr, int index = 0)
{
if (index < 0 || index >= NumItems || !IsDataAPointer) return default;
// why is twain being difficult and not use TW_STR* like a normal person.
// what even is the encoding for those things? Imma yolo it.
string? value;
var itemAsPtr = (IntPtr)Item.ToPointer(); // this is also iffy
if (NumItems == 1)
{
// if 1, item is already the pointer to the string
value = LockAndReadNullTerminatedString(memMgr, itemAsPtr);
}
else
{
// if more than 1, item points to an array of pointers that each points to their own string
var lockPtr = memMgr.Lock(itemAsPtr);
lockPtr += (IntPtr.Size * index);
// is this even correct? I hope it is
var subItemPtr = Marshal.PtrToStructure<IntPtr>(lockPtr);
value = LockAndReadNullTerminatedString(memMgr, subItemPtr);
memMgr.Unlock(itemAsPtr);
}
return value;
}
private string? LockAndReadNullTerminatedString(IMemoryManager memMgr, IntPtr data)
{
var lockPtr = memMgr.Lock(data);
// yolo as ansi, should work in most cases
var value = Marshal.PtrToStringAnsi(lockPtr);
memMgr.Unlock(data);
return value;
}
/// <summary>
/// Frees all DS-allocated memory if necessary.
/// </summary> /// </summary>
/// <param name="memMgr"></param> /// <param name="memMgr"></param>
internal unsafe void Free(IMemoryManager memMgr) internal unsafe void Free(IMemoryManager memMgr)
{ {
if (ReturnCode == TWRC.SUCCESS && Item != UIntPtr.Zero && if (ReturnCode != TWRC.SUCCESS || !IsDataAPointer) return;
(ItemType == TWTY.HANDLE || (ItemType.GetItemTypeSize() * NumItems) > 4))
var itemAsPtr = (IntPtr)Item.ToPointer(); // this is also iffy
if (ItemType == TWTY.HANDLE && NumItems > 1)
{ {
// this is probably not correct // must go into each handle in the array and free them individually :(
memMgr.Free((IntPtr)Item.ToPointer()); var lockPtr = memMgr.Lock(itemAsPtr);
Item = UIntPtr.Zero; // this is only a copy but whatev for (var i = 0; i < NumItems; i++)
{
// is this even correct? I hope it is
var subItemPtr = Marshal.PtrToStructure<IntPtr>(lockPtr);
memMgr.Free(subItemPtr);
lockPtr += IntPtr.Size;
}
} }
memMgr.Free(itemAsPtr);
Item = UIntPtr.Zero;
} }
} }
} }

View File

@ -430,7 +430,7 @@ namespace NTwain.Data
/// <param name="type">The twain type.</param> /// <param name="type">The twain type.</param>
/// <param name="itemIndex">Index of the item if pointer is array.</param> /// <param name="itemIndex">Index of the item if pointer is array.</param>
/// <returns></returns> /// <returns></returns>
static TValue ReadTWTYData<TValue>(this IntPtr intptr, TWTY type, int itemIndex) where TValue : struct public static TValue ReadTWTYData<TValue>(this IntPtr intptr, TWTY type, int itemIndex) where TValue : struct
{ {
var isEnum = typeof(TValue).IsEnum; var isEnum = typeof(TValue).IsEnum;