392 lines
12 KiB
C
392 lines
12 KiB
C
/**
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
* RemoteFX USB Redirection
|
|
*
|
|
* Copyright 2012 Atrust corp.
|
|
* Copyright 2012 Alfred Liu <alfred.liu@atruscorp.com>
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <freerdp/log.h>
|
|
#include <freerdp/utils/msusb.h>
|
|
|
|
#define TAG FREERDP_TAG("utils")
|
|
|
|
#ifdef WITH_DEBUG_MSUSB
|
|
#define DEBUG_MSUSB(...) WLog_DBG(TAG, __VA_ARGS__)
|
|
#else
|
|
#define DEBUG_MSUSB(...) do { } while (0)
|
|
#endif
|
|
|
|
|
|
static MSUSB_PIPE_DESCRIPTOR* msusb_mspipe_new()
|
|
{
|
|
return (MSUSB_PIPE_DESCRIPTOR*) calloc(1, sizeof(MSUSB_PIPE_DESCRIPTOR));
|
|
}
|
|
|
|
static void msusb_mspipes_free(MSUSB_PIPE_DESCRIPTOR** MsPipes,
|
|
UINT32 NumberOfPipes)
|
|
{
|
|
UINT32 pnum = 0;
|
|
|
|
if (MsPipes)
|
|
{
|
|
for (pnum = 0; pnum < NumberOfPipes && MsPipes[pnum]; pnum++)
|
|
{
|
|
zfree(MsPipes[pnum]);
|
|
}
|
|
|
|
zfree(MsPipes);
|
|
}
|
|
}
|
|
|
|
void msusb_mspipes_replace(MSUSB_INTERFACE_DESCRIPTOR* MsInterface,
|
|
MSUSB_PIPE_DESCRIPTOR** NewMsPipes, UINT32 NewNumberOfPipes)
|
|
{
|
|
/* free orignal MsPipes */
|
|
msusb_mspipes_free(MsInterface->MsPipes, MsInterface->NumberOfPipes);
|
|
/* And replace it */
|
|
MsInterface->MsPipes = NewMsPipes;
|
|
MsInterface->NumberOfPipes = NewNumberOfPipes;
|
|
}
|
|
|
|
static MSUSB_PIPE_DESCRIPTOR** msusb_mspipes_read(BYTE* data, UINT32 data_size,
|
|
UINT32 NumberOfPipes, int* offset)
|
|
{
|
|
UINT32 pnum, move = 0;
|
|
MSUSB_PIPE_DESCRIPTOR** MsPipes;
|
|
MsPipes = (MSUSB_PIPE_DESCRIPTOR**) calloc(NumberOfPipes,
|
|
sizeof(MSUSB_PIPE_DESCRIPTOR*));
|
|
|
|
if (!MsPipes)
|
|
return NULL;
|
|
|
|
for (pnum = 0; pnum < NumberOfPipes; pnum++)
|
|
{
|
|
MSUSB_PIPE_DESCRIPTOR* MsPipe = msusb_mspipe_new();
|
|
|
|
if (!MsPipe)
|
|
goto out_error;
|
|
|
|
data_read_UINT16(data + move, MsPipe->MaximumPacketSize);
|
|
data_read_UINT32(data + move + 4, MsPipe->MaximumTransferSize);
|
|
data_read_UINT32(data + move + 8, MsPipe->PipeFlags);
|
|
move += 12;
|
|
/* Already set to zero by memset
|
|
MsPipe->PipeHandle = 0;
|
|
MsPipe->bEndpointAddress = 0;
|
|
MsPipe->bInterval = 0;
|
|
MsPipe->PipeType = 0;
|
|
MsPipe->InitCompleted = 0;
|
|
*/
|
|
MsPipes[pnum] = MsPipe;
|
|
}
|
|
|
|
*offset += move;
|
|
return MsPipes;
|
|
out_error:
|
|
|
|
for (pnum = 0; pnum < NumberOfPipes; pnum++)
|
|
{
|
|
free(MsPipes[pnum]);
|
|
}
|
|
|
|
free(MsPipes);
|
|
return NULL;
|
|
}
|
|
|
|
static MSUSB_INTERFACE_DESCRIPTOR* msusb_msinterface_new()
|
|
{
|
|
return (MSUSB_INTERFACE_DESCRIPTOR*) calloc(1,
|
|
sizeof(MSUSB_INTERFACE_DESCRIPTOR));
|
|
}
|
|
|
|
static void msusb_msinterface_free(MSUSB_INTERFACE_DESCRIPTOR* MsInterface)
|
|
{
|
|
if (MsInterface)
|
|
{
|
|
msusb_mspipes_free(MsInterface->MsPipes, MsInterface->NumberOfPipes);
|
|
MsInterface->MsPipes = NULL;
|
|
zfree(MsInterface);
|
|
}
|
|
}
|
|
|
|
static void msusb_msinterface_free_list(MSUSB_INTERFACE_DESCRIPTOR**
|
|
MsInterfaces, UINT32 NumInterfaces)
|
|
{
|
|
UINT32 inum = 0;
|
|
|
|
if (MsInterfaces)
|
|
{
|
|
for (inum = 0; inum < NumInterfaces; inum++)
|
|
{
|
|
msusb_msinterface_free(MsInterfaces[inum]);
|
|
}
|
|
|
|
zfree(MsInterfaces);
|
|
}
|
|
}
|
|
|
|
void msusb_msinterface_replace(MSUSB_CONFIG_DESCRIPTOR* MsConfig,
|
|
BYTE InterfaceNumber, MSUSB_INTERFACE_DESCRIPTOR* NewMsInterface)
|
|
{
|
|
msusb_msinterface_free(MsConfig->MsInterfaces[InterfaceNumber]);
|
|
MsConfig->MsInterfaces[InterfaceNumber] = NewMsInterface;
|
|
}
|
|
|
|
MSUSB_INTERFACE_DESCRIPTOR* msusb_msinterface_read(BYTE* data, UINT32 data_size,
|
|
int* offset)
|
|
{
|
|
MSUSB_INTERFACE_DESCRIPTOR* MsInterface;
|
|
MsInterface = msusb_msinterface_new();
|
|
|
|
if (!MsInterface)
|
|
return NULL;
|
|
|
|
data_read_UINT16(data, MsInterface->Length);
|
|
data_read_UINT16(data + 2, MsInterface->NumberOfPipesExpected);
|
|
data_read_BYTE(data + 4, MsInterface->InterfaceNumber);
|
|
data_read_BYTE(data + 5, MsInterface->AlternateSetting);
|
|
data_read_UINT32(data + 8, MsInterface->NumberOfPipes);
|
|
*offset += 12;
|
|
MsInterface->InterfaceHandle = 0;
|
|
MsInterface->bInterfaceClass = 0;
|
|
MsInterface->bInterfaceSubClass = 0;
|
|
MsInterface->bInterfaceProtocol = 0;
|
|
MsInterface->InitCompleted = 0;
|
|
MsInterface->MsPipes = NULL;
|
|
|
|
if (MsInterface->NumberOfPipes > 0)
|
|
{
|
|
MsInterface->MsPipes =
|
|
msusb_mspipes_read(data + (*offset), data_size - (*offset),
|
|
MsInterface->NumberOfPipes, offset);
|
|
|
|
if (!MsInterface->MsPipes)
|
|
goto out_error;
|
|
}
|
|
|
|
return MsInterface;
|
|
out_error:
|
|
msusb_msinterface_free(MsInterface);
|
|
return NULL;
|
|
}
|
|
|
|
int msusb_msinterface_write(MSUSB_INTERFACE_DESCRIPTOR* MsInterface, BYTE* data,
|
|
int* offset)
|
|
{
|
|
MSUSB_PIPE_DESCRIPTOR** MsPipes;
|
|
MSUSB_PIPE_DESCRIPTOR* MsPipe;
|
|
UINT32 pnum = 0, move = 0;
|
|
/* Length */
|
|
data_write_UINT16(data, MsInterface->Length);
|
|
/* InterfaceNumber */
|
|
data_write_BYTE(data + 2, MsInterface->InterfaceNumber);
|
|
/* AlternateSetting */
|
|
data_write_BYTE(data + 3, MsInterface->AlternateSetting);
|
|
/* bInterfaceClass */
|
|
data_write_BYTE(data + 4, MsInterface->bInterfaceClass);
|
|
/* bInterfaceSubClass */
|
|
data_write_BYTE(data + 5, MsInterface->bInterfaceSubClass);
|
|
/* bInterfaceProtocol */
|
|
data_write_BYTE(data + 6, MsInterface->bInterfaceProtocol);
|
|
/* Padding */
|
|
data_write_BYTE(data + 7, 0);
|
|
/* InterfaceHandle */
|
|
data_write_UINT32(data + 8, MsInterface->InterfaceHandle);
|
|
/* NumberOfPipes */
|
|
data_write_UINT32(data + 12, MsInterface->NumberOfPipes);
|
|
move += 16;
|
|
/* Pipes */
|
|
MsPipes = MsInterface->MsPipes;
|
|
|
|
for (pnum = 0; pnum < MsInterface->NumberOfPipes; pnum++)
|
|
{
|
|
MsPipe = MsPipes[pnum];
|
|
/* MaximumPacketSize */
|
|
data_write_UINT16(data + move, MsPipe->MaximumPacketSize);
|
|
/* EndpointAddress */
|
|
data_write_BYTE(data + move + 2, MsPipe->bEndpointAddress);
|
|
/* Interval */
|
|
data_write_BYTE(data + move + 3, MsPipe->bInterval);
|
|
/* PipeType */
|
|
data_write_UINT32(data + move + 4, MsPipe->PipeType);
|
|
/* PipeHandle */
|
|
data_write_UINT32(data + move + 8, MsPipe->PipeHandle);
|
|
/* MaximumTransferSize */
|
|
data_write_UINT32(data + move + 12, MsPipe->MaximumTransferSize);
|
|
/* PipeFlags */
|
|
data_write_UINT32(data + move + 16, MsPipe->PipeFlags);
|
|
move += 20;
|
|
}
|
|
|
|
*offset += move;
|
|
return 0;
|
|
}
|
|
|
|
static MSUSB_INTERFACE_DESCRIPTOR** msusb_msinterface_read_list(BYTE* data,
|
|
UINT32 data_size, UINT32 NumInterfaces)
|
|
{
|
|
UINT32 inum;
|
|
int offset = 0;
|
|
MSUSB_INTERFACE_DESCRIPTOR** MsInterfaces;
|
|
MsInterfaces = (MSUSB_INTERFACE_DESCRIPTOR**) calloc(NumInterfaces,
|
|
sizeof(MSUSB_INTERFACE_DESCRIPTOR*));
|
|
|
|
if (!MsInterfaces)
|
|
return NULL;
|
|
|
|
for (inum = 0; inum < NumInterfaces; inum++)
|
|
{
|
|
MsInterfaces[inum] = msusb_msinterface_read(data + offset, data_size - offset,
|
|
&offset);
|
|
}
|
|
|
|
return MsInterfaces;
|
|
}
|
|
|
|
int msusb_msconfig_write(MSUSB_CONFIG_DESCRIPTOR* MsConfg, BYTE* data,
|
|
int* offset)
|
|
{
|
|
UINT32 inum = 0;
|
|
MSUSB_INTERFACE_DESCRIPTOR** MsInterfaces;
|
|
MSUSB_INTERFACE_DESCRIPTOR* MsInterface;
|
|
/* ConfigurationHandle*/
|
|
data_write_UINT32(data + *offset, MsConfg->ConfigurationHandle);
|
|
/* NumInterfaces*/
|
|
data_write_UINT32(data + *offset + 4, MsConfg->NumInterfaces);
|
|
*offset += 8;
|
|
/* Interfaces */
|
|
MsInterfaces = MsConfg->MsInterfaces;
|
|
|
|
for (inum = 0; inum < MsConfg->NumInterfaces; inum++)
|
|
{
|
|
MsInterface = MsInterfaces[inum];
|
|
msusb_msinterface_write(MsInterface, data + (*offset), offset);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
MSUSB_CONFIG_DESCRIPTOR* msusb_msconfig_new()
|
|
{
|
|
return (MSUSB_CONFIG_DESCRIPTOR*) calloc(1, sizeof(MSUSB_CONFIG_DESCRIPTOR));
|
|
}
|
|
|
|
void msusb_msconfig_free(MSUSB_CONFIG_DESCRIPTOR* MsConfig)
|
|
{
|
|
if (MsConfig)
|
|
{
|
|
msusb_msinterface_free_list(MsConfig->MsInterfaces, MsConfig->NumInterfaces);
|
|
MsConfig->MsInterfaces = NULL;
|
|
zfree(MsConfig);
|
|
}
|
|
}
|
|
|
|
MSUSB_CONFIG_DESCRIPTOR* msusb_msconfig_read(BYTE* data, UINT32 data_size,
|
|
UINT32 NumInterfaces)
|
|
{
|
|
UINT32 i, offset = 0;
|
|
UINT16 lenInterface;
|
|
MSUSB_CONFIG_DESCRIPTOR* MsConfig;
|
|
BYTE lenConfiguration, typeConfiguration;
|
|
MsConfig = msusb_msconfig_new();
|
|
|
|
for (i = 0; i < NumInterfaces; i++)
|
|
{
|
|
data_read_UINT16(data + offset, lenInterface);
|
|
offset += lenInterface;
|
|
}
|
|
|
|
data_read_BYTE(data + offset, lenConfiguration);
|
|
data_read_BYTE(data + offset + 1, typeConfiguration);
|
|
|
|
if (lenConfiguration != 0x9 || typeConfiguration != 0x2)
|
|
{
|
|
DEBUG_MSUSB("%s: len and type must be 0x9 and 0x2 , but it is 0x%"PRIx8" and 0x%"PRIx8"",
|
|
__FUNCTION__, lenConfiguration, typeConfiguration);
|
|
}
|
|
|
|
data_read_UINT16(data + offset + 2, MsConfig->wTotalLength);
|
|
data_read_BYTE(data + offset + 5, MsConfig->bConfigurationValue);
|
|
MsConfig->NumInterfaces = NumInterfaces;
|
|
MsConfig->ConfigurationHandle = 0;
|
|
MsConfig->InitCompleted = 0;
|
|
MsConfig->MsOutSize = 0;
|
|
MsConfig->MsInterfaces = NULL;
|
|
offset = 0;
|
|
|
|
if (NumInterfaces > 0)
|
|
{
|
|
MsConfig->MsInterfaces = msusb_msinterface_read_list(data, data_size,
|
|
NumInterfaces);
|
|
}
|
|
|
|
return MsConfig;
|
|
}
|
|
|
|
void msusb_msconfig_dump(MSUSB_CONFIG_DESCRIPTOR* MsConfig)
|
|
{
|
|
MSUSB_INTERFACE_DESCRIPTOR** MsInterfaces;
|
|
MSUSB_INTERFACE_DESCRIPTOR* MsInterface;
|
|
MSUSB_PIPE_DESCRIPTOR** MsPipes;
|
|
MSUSB_PIPE_DESCRIPTOR* MsPipe;
|
|
UINT32 inum = 0, pnum = 0;
|
|
WLog_INFO(TAG, "=================MsConfig:========================");
|
|
WLog_INFO(TAG, "wTotalLength:%"PRIu16"", MsConfig->wTotalLength);
|
|
WLog_INFO(TAG, "bConfigurationValue:%"PRIu8"", MsConfig->bConfigurationValue);
|
|
WLog_INFO(TAG, "ConfigurationHandle:0x%08"PRIx32"", MsConfig->ConfigurationHandle);
|
|
WLog_INFO(TAG, "InitCompleted:%d", MsConfig->InitCompleted);
|
|
WLog_INFO(TAG, "MsOutSize:%d", MsConfig->MsOutSize);
|
|
WLog_INFO(TAG, "NumInterfaces:%"PRIu32"", MsConfig->NumInterfaces);
|
|
MsInterfaces = MsConfig->MsInterfaces;
|
|
|
|
for (inum = 0; inum < MsConfig->NumInterfaces; inum++)
|
|
{
|
|
MsInterface = MsInterfaces[inum];
|
|
WLog_INFO(TAG, " Interface: %"PRIu8"", MsInterface->InterfaceNumber);
|
|
WLog_INFO(TAG, " Length: %"PRIu16"", MsInterface->Length);
|
|
WLog_INFO(TAG, " NumberOfPipesExpected: %"PRIu16"",
|
|
MsInterface->NumberOfPipesExpected);
|
|
WLog_INFO(TAG, " AlternateSetting: %"PRIu8"", MsInterface->AlternateSetting);
|
|
WLog_INFO(TAG, " NumberOfPipes: %"PRIu32"", MsInterface->NumberOfPipes);
|
|
WLog_INFO(TAG, " InterfaceHandle: 0x%08"PRIx32"", MsInterface->InterfaceHandle);
|
|
WLog_INFO(TAG, " bInterfaceClass: 0x%02"PRIx8"", MsInterface->bInterfaceClass);
|
|
WLog_INFO(TAG, " bInterfaceSubClass: 0x%02"PRIx8"", MsInterface->bInterfaceSubClass);
|
|
WLog_INFO(TAG, " bInterfaceProtocol: 0x%02"PRIx8"", MsInterface->bInterfaceProtocol);
|
|
WLog_INFO(TAG, " InitCompleted: %d", MsInterface->InitCompleted);
|
|
MsPipes = MsInterface->MsPipes;
|
|
|
|
for (pnum = 0; pnum < MsInterface->NumberOfPipes; pnum++)
|
|
{
|
|
MsPipe = MsPipes[pnum];
|
|
WLog_INFO(TAG, " Pipe: %d", pnum);
|
|
WLog_INFO(TAG, " MaximumPacketSize: 0x%04"PRIx16"", MsPipe->MaximumPacketSize);
|
|
WLog_INFO(TAG, " MaximumTransferSize: 0x%08"PRIx32"", MsPipe->MaximumTransferSize);
|
|
WLog_INFO(TAG, " PipeFlags: 0x%08"PRIx32"", MsPipe->PipeFlags);
|
|
WLog_INFO(TAG, " PipeHandle: 0x%08"PRIx32"", MsPipe->PipeHandle);
|
|
WLog_INFO(TAG, " bEndpointAddress: 0x%02"PRIx8"", MsPipe->bEndpointAddress);
|
|
WLog_INFO(TAG, " bInterval: %"PRIu8"", MsPipe->bInterval);
|
|
WLog_INFO(TAG, " PipeType: 0x%02"PRIx8"", MsPipe->PipeType);
|
|
WLog_INFO(TAG, " InitCompleted: %d", MsPipe->InitCompleted);
|
|
}
|
|
}
|
|
|
|
WLog_INFO(TAG, "==================================================");
|
|
}
|