#region Copyright(c) Travis Robinson
#endregion
#define BMFW
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Runtime.InteropServices;
using Test.Devices;
using libusbK;
using libusbK.Examples;
namespace Xfer.Async
{
internal class Program
{
#region TODO USER: Set the test parameters for your device.
public static AsyncTestParameters Test = new AsyncTestParameters(0x04d8, 0xfa2e, 0, 0x81, 9, null, -1, 3);
#endregion
private static void Main()
{
bool success;
WINUSB_PIPE_INFORMATION_EX pipeInfo;
UsbK usb;
USB_INTERFACE_DESCRIPTOR interfaceDescriptor;
if (!Test.ConfigureDevice(out pipeInfo, out usb, out interfaceDescriptor)) return;
if (Test.TransferBufferSize == -1)
Test.TransferBufferSize = pipeInfo.MaximumPacketSize * 64;
#if BMFW
Console.WriteLine("Configuring for benchmark device..");
BM_TEST_TYPE testType = ((Test.PipeId & 0x80) > 0) ? BM_TEST_TYPE.READ : BM_TEST_TYPE.WRITE;
success = Benchmark.Configure(usb, BM_COMMAND.SET_TEST, interfaceDescriptor.bInterfaceNumber, ref testType);
if (!success)
{
Console.WriteLine("Bench_Configure failed.");
}
#endif
if (!Test.ShowTestReady()) goto Done;
int[] pipeTimeoutMS = new[] {0};
usb.SetPipePolicy((byte) Test.PipeId,
(uint) PipePolicyType.PIPE_TRANSFER_TIMEOUT,
(uint) Marshal.SizeOf(typeof (int)),
pipeTimeoutMS);
int[] useRawIO = new[] {1};
usb.SetPipePolicy((byte)Test.PipeId,
(uint)PipePolicyType.RAW_IO,
(uint) Marshal.SizeOf(typeof(int)),
useRawIO);
int totalSubmittedTransfers = 0;
int totalCompletedTransfers = 0;
byte[][] transferBuffers = new byte[Test.MaxPendingIO][];
OvlK ovlPool = new OvlK(usb.Handle, Test.MaxPendingIO, KOVL_POOL_FLAG.NONE);
List<GCHandle> transferBufferGCList = new List<GCHandle>(Test.MaxPendingIO);
success = true;
while (success && totalCompletedTransfers < Test.MaxTransfersTotal)
{
KOVL_HANDLE ovlHandle;
uint transferred;
while (success && totalSubmittedTransfers < Test.MaxTransfersTotal)
{
if (!ovlPool.Acquire(out ovlHandle))
{
Debug.Assert(Marshal.GetLastWin32Error() == ErrorCodes.NoMoreItems);
break;
}
int transferBufferIndex = totalSubmittedTransfers % Test.MaxPendingIO;
if (transferBuffers[transferBufferIndex] == null)
{
Console.WriteLine("Allocating transfer buffer at index:{0}", transferBufferIndex);
transferBuffers[transferBufferIndex] = new byte[Test.TransferBufferSize];
transferBufferGCList.Add(GCHandle.Alloc(transferBuffers[transferBufferIndex], GCHandleType.Pinned));
}
byte[] transferBuffer = transferBuffers[transferBufferIndex];
uint not_used_for_async;
if ((Test.PipeId & AllKConstants.USB_ENDPOINT_DIRECTION_MASK) > 0)
{
success = usb.ReadPipe((byte) Test.PipeId, transferBuffer, (uint)transferBuffer.Length, out not_used_for_async, ovlHandle);
}
else
{
FillMyBufferForWrite(transferBuffer, out transferred);
success = usb.WritePipe((byte)Test.PipeId, transferBuffer, (uint)transferred, out not_used_for_async, ovlHandle);
}
if (Marshal.GetLastWin32Error() == ErrorCodes.IoPending)
{
success = true;
totalSubmittedTransfers++;
Console.WriteLine("Pending #{0:0000} {1} bytes.", totalSubmittedTransfers, transferBuffer.Length);
}
else
Console.WriteLine("Pending #{0:0000} failed. ErrorCode={1:X8}h", totalSubmittedTransfers, Marshal.GetLastWin32Error());
}
if (!success) break;
success = ovlPool.WaitOldest(out ovlHandle, 1000, KOVL_WAIT_FLAG.RELEASE_ALWAYS, out transferred);
totalCompletedTransfers++;
if (success)
{
if ((Test.PipeId & AllKConstants.USB_ENDPOINT_DIRECTION_MASK) > 0)
{
ProcessMyBufferFromRead(transferBuffers[(totalCompletedTransfers-1) % Test.MaxPendingIO], transferred);
}
Console.WriteLine("Complete #{0:0000} {1} bytes.", totalCompletedTransfers, transferred);
}
else
Console.WriteLine("Complete #{0:0000} Wait failed. ErrorCode={1:X8}h", totalCompletedTransfers, Marshal.GetLastWin32Error());
}
if (!success)
Console.WriteLine("An error occured transferring data. ErrorCode: {0:X8}h", Marshal.GetLastWin32Error());
ovlPool.Free();
foreach (GCHandle gcHandle in transferBufferGCList)
gcHandle.Free();
Done:
usb.Free();
}
#region TODO USER: Use these functions to process and fill the transfer buffers
private static void ProcessMyBufferFromRead(byte[] transferBuffer, uint transferred)
{
}
private static void FillMyBufferForWrite(byte[] transferBuffer, out uint length)
{
length = (uint) transferBuffer.Length;
}
#endregion
}
}