avrdude/external/libusb/src/windows.cpp

470 lines
11 KiB
C++

/*
* avrdude - A Downloader/Uploader for AVR device programmers
* Copyright (C) 2019 Marius Greuel
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "libwinusb.h"
extern "C"
{
#include "error.h"
#include "usbi.h"
}
static std::string CreateMessageFromError(HRESULT hr)
{
LPSTR pszMessage = nullptr;
DWORD dwChars = FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, nullptr, hr, 0, (LPSTR)&pszMessage, 0, nullptr);
if (dwChars > 0 && pszMessage != nullptr)
{
if (dwChars > 0 && pszMessage[dwChars - 1] == '\n')
dwChars--;
if (dwChars > 0 && pszMessage[dwChars - 1] == '\r')
dwChars--;
std::string message = std::string(pszMessage, dwChars);
LocalFree(pszMessage);
return message;
}
else
{
return "Unknown error";
}
}
static int usb_hresult_to_errno(HRESULT hr)
{
if (SUCCEEDED(hr))
return 0;
switch (hr)
{
case HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER):
return EINVAL;
case HRESULT_FROM_WIN32(ERROR_NOT_ENOUGH_MEMORY):
return ENOMEM;
case HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED):
return EPERM;
case HRESULT_FROM_WIN32(ERROR_INVALID_HANDLE):
case HRESULT_FROM_WIN32(ERROR_BAD_DEVICE):
case HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND):
return ENODEV;
case HRESULT_FROM_WIN32(ERROR_SEM_TIMEOUT):
case HRESULT_FROM_WIN32(ERROR_OPERATION_ABORTED):
return ETIMEDOUT;
default:
return EIO;
}
}
void usb_os_init(void)
{
}
void usb_set_debug(int level)
{
if (usb_log_get_level() || level)
{
USBMSG("setting debugging level to %d (%s)\n", level, level ? "on" : "off");
}
usb_log_set_level(static_cast<USB_LOG_LEVEL>(level));
}
int usb_os_find_busses(struct usb_bus** busses)
{
auto bus = static_cast<struct usb_bus*>(std::malloc(sizeof(struct usb_bus)));
if (bus == nullptr)
{
USBERR("memory allocation failed\n");
return -ENOMEM;
}
std::memset(bus, 0, sizeof(*bus));
std::strcpy(bus->dirname, "bus-0");
USBMSG("found %s\n", bus->dirname);
*busses = bus;
return 0;
}
int usb_os_find_devices(struct usb_bus* bus, struct usb_device** devices)
{
if (bus == nullptr)
{
return -EINVAL;
}
if (devices == nullptr)
{
return -EINVAL;
}
LibWinUsb::UsbDeviceEnumerator enumerator;
HRESULT hr = enumerator.EnumerateDevices();
if (FAILED(hr))
{
return -1;
}
struct usb_device* fdev = nullptr;
const auto& list = enumerator.GetDevices();
for (const auto& item : list)
{
struct usb_device* dev = static_cast<struct usb_device*>(std::malloc(sizeof(struct usb_device)));
if (dev == nullptr)
{
USBERR0("memory allocation failed\n");
return -ENOMEM;
}
std::memset(dev, 0, sizeof(*dev));
std::unique_ptr<LibWinUsb::UsbDriver> driver;
if (FAILED(item->Open(driver)) ||
FAILED(driver->GetDescriptor(USB_DT_DEVICE, 0, 0, &dev->descriptor, sizeof(dev->descriptor), nullptr, 5000)))
{
free(dev);
continue;
}
_snprintf(dev->filename, LIBUSB_PATH_MAX - 1, "%s--%s",
LibWinUsb::Encoding::ToAnsi(item->GetInterfacePath()).c_str(),
LibWinUsb::Encoding::ToAnsi(item->GetService()).c_str());
LIST_ADD(fdev, dev);
USBMSG("found %s on %s\n", dev->filename, bus->dirname);
}
*devices = fdev;
return 0;
}
int usb_os_determine_children(struct usb_bus* bus)
{
return 0;
}
int usb_os_open(usb_dev_handle* dev)
{
if (dev == nullptr)
{
USBERR("invalid device handle %p", dev);
return -EINVAL;
}
dev->impl_info = nullptr;
dev->config = 0;
dev->interface = -1;
dev->altsetting = -1;
std::string dev_name = dev->device->filename;
auto pos = dev_name.find("--");
if (pos == std::string::npos)
{
USBERR("invalid file name %s\n", dev->device->filename);
return -ENOENT;
}
std::unique_ptr<LibWinUsb::UsbDriver> driver;
HRESULT hr = LibWinUsb::UsbDevice::Open(LibWinUsb::Encoding::ToUnicode(dev_name.substr(pos + 2)), LibWinUsb::Encoding::ToUnicode(dev_name.substr(0, pos)), driver);
if (FAILED(hr))
{
USBERR("failed to open %s: win error: %s\n", dev->device->filename, CreateMessageFromError(hr).c_str());
return -ENOENT;
}
dev->impl_info = driver.release();
return 0;
}
int usb_os_close(usb_dev_handle* dev)
{
delete static_cast<LibWinUsb::UsbDriver*>(dev->impl_info);
dev->impl_info = nullptr;
dev->interface = -1;
dev->altsetting = -1;
return 0;
}
int usb_control_msg(usb_dev_handle* dev, int requestType, int request, int value, int index, char* bytes, int size, int timeout)
{
if (dev->impl_info == nullptr)
{
USBERR0("error: device not open\n");
return -EINVAL;
}
if (requestType == (USB_ENDPOINT_IN | USB_TYPE_STANDARD | USB_RECIP_DEVICE) && request == USB_REQ_GET_DESCRIPTOR)
{
uint32_t bytesReturned = 0;
HRESULT hr = static_cast<LibWinUsb::UsbDriver*>(dev->impl_info)->GetDescriptor(
static_cast<uint8_t>(value >> 8),
static_cast<uint8_t>(value),
static_cast<uint16_t>(index),
bytes,
static_cast<uint16_t>(size),
&bytesReturned,
timeout);
if (FAILED(hr))
{
return -usb_hresult_to_errno(hr);
}
return static_cast<int>(bytesReturned);
}
else
{
uint32_t bytesReturned = 0;
HRESULT hr = static_cast<LibWinUsb::UsbDriver*>(dev->impl_info)->ControlTransfer(
static_cast<uint8_t>(requestType),
static_cast<uint8_t>(request),
static_cast<uint16_t>(value),
static_cast<uint16_t>(index),
bytes,
static_cast<uint16_t>(size),
&bytesReturned,
timeout);
if (FAILED(hr))
{
return -usb_hresult_to_errno(hr);
}
return static_cast<int>(bytesReturned);
}
}
int usb_set_configuration(usb_dev_handle* dev, int configuration)
{
if (dev->impl_info == nullptr)
{
USBERR0("error: device not open\n");
return -EINVAL;
}
if (dev->config == configuration)
{
return 0;
}
if (dev->interface >= 0)
{
USBERR0("can't change configuration, an interface is still in use (claimed)\n");
return -EINVAL;
}
HRESULT hr = static_cast<LibWinUsb::UsbDriver*>(dev->impl_info)->SetConfiguration(static_cast<uint8_t>(configuration));
if (FAILED(hr))
{
return -usb_hresult_to_errno(hr);
}
dev->config = configuration;
dev->interface = -1;
dev->altsetting = -1;
return 0;
}
int usb_claim_interface(usb_dev_handle* dev, int interface)
{
if (dev->impl_info == nullptr)
{
USBERR0("device not open\n");
return -EINVAL;
}
if (dev->config == 0)
{
USBERR("could not claim interface %d, invalid configuration %d\n", interface, dev->config);
return -EINVAL;
}
if (dev->interface == interface)
{
return 0;
}
HRESULT hr = static_cast<LibWinUsb::UsbDriver*>(dev->impl_info)->ClaimInterface(static_cast<uint8_t>(interface));
if (FAILED(hr))
{
return -usb_hresult_to_errno(hr);
}
dev->interface = interface;
dev->altsetting = 0;
return 0;
}
int usb_release_interface(usb_dev_handle* dev, int interface)
{
if (dev->impl_info == nullptr)
{
USBERR0("device not open\n");
return -EINVAL;
}
if (dev->config == 0)
{
USBERR("could not release interface %d, invalid configuration %d\n", interface, dev->config);
return -EINVAL;
}
HRESULT hr = static_cast<LibWinUsb::UsbDriver*>(dev->impl_info)->ReleaseInterface(static_cast<uint8_t>(interface));
if (FAILED(hr))
{
return -usb_hresult_to_errno(hr);
}
dev->interface = -1;
dev->altsetting = -1;
return 0;
}
int usb_set_altinterface(usb_dev_handle* dev, int alternate)
{
if (dev->impl_info == nullptr)
{
USBERR0("device not open\n");
return -EINVAL;
}
if (dev->config == 0)
{
USBERR("could not set alt interface %d: invalid configuration %d\n", alternate, dev->config);
return -EINVAL;
}
if (dev->interface < 0)
{
USBERR("could not set alt interface %d: no interface claimed\n", alternate);
return -EINVAL;
}
HRESULT hr = static_cast<LibWinUsb::UsbDriver*>(dev->impl_info)->SetAlternateSetting(
static_cast<uint8_t>(dev->interface),
static_cast<uint8_t>(alternate));
if (FAILED(hr))
{
return -usb_hresult_to_errno(hr);
}
dev->altsetting = alternate;
return 0;
}
int usb_bulk_read(usb_dev_handle* dev, int ep, char* bytes, int size, int timeout)
{
if (dev->impl_info == nullptr)
{
USBERR0("device not open\n");
return -EINVAL;
}
if (dev->config == 0)
{
USBERR("invalid configuration %d\n", dev->config);
return -EINVAL;
}
if (dev->interface < 0)
{
USBERR("invalid interface %d\n", dev->interface);
return -EINVAL;
}
if ((ep & USB_ENDPOINT_IN) == 0)
{
USBERR("invalid endpoint 0x%02x", ep);
return -EINVAL;
}
uint32_t bytesRead = 0;
HRESULT hr = static_cast<LibWinUsb::UsbDriver*>(dev->impl_info)->ReadPipe(
static_cast<uint8_t>(ep),
bytes,
size,
&bytesRead,
timeout);
if (FAILED(hr))
{
return -usb_hresult_to_errno(hr);
}
return static_cast<int>(bytesRead);
}
int usb_bulk_write(usb_dev_handle* dev, int ep, char* bytes, int size, int timeout)
{
if (dev->impl_info == nullptr)
{
USBERR0("device not open\n");
return -EINVAL;
}
if (dev->config == 0)
{
USBERR("invalid configuration %d\n", dev->config);
return -EINVAL;
}
if (dev->interface < 0)
{
USBERR("invalid interface %d\n", dev->interface);
return -EINVAL;
}
if ((ep & USB_ENDPOINT_IN) != 0)
{
USBERR("invalid endpoint 0x%02x", ep);
return -EINVAL;
}
uint32_t bytesWritten = 0;
HRESULT hr = static_cast<LibWinUsb::UsbDriver*>(dev->impl_info)->WritePipe(
static_cast<uint8_t>(ep),
bytes,
size,
&bytesWritten,
timeout);
if (FAILED(hr))
{
return -usb_hresult_to_errno(hr);
}
return static_cast<int>(bytesWritten);
}
int usb_interrupt_read(usb_dev_handle* dev, int ep, char* bytes, int size, int timeout)
{
return usb_bulk_read(dev, ep, bytes, size, timeout);
}
int usb_interrupt_write(usb_dev_handle* dev, int ep, char* bytes, int size, int timeout)
{
return usb_bulk_write(dev, ep, bytes, size, timeout);
}