2021-02-27 15:10:37 +11:00
|
|
|
/*
|
|
|
|
|
* Copyright (c) 2021, Jesse Buhagiar <jooster669@gmail.com>
|
|
|
|
|
*
|
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
|
*/
|
|
|
|
|
|
2021-06-25 09:51:22 +03:00
|
|
|
#include <Kernel/Bus/USB/PacketTypes.h>
|
2021-08-13 16:10:43 +10:00
|
|
|
#include <Kernel/Bus/USB/UHCI/UHCIController.h>
|
2021-06-25 09:51:22 +03:00
|
|
|
#include <Kernel/Bus/USB/USBPipe.h>
|
|
|
|
|
#include <Kernel/Bus/USB/USBTransfer.h>
|
2021-02-27 15:10:37 +11:00
|
|
|
|
|
|
|
|
namespace Kernel::USB {
|
|
|
|
|
|
2021-08-08 19:50:20 +01:00
|
|
|
KResultOr<NonnullOwnPtr<Pipe>> Pipe::try_create_pipe(USBController const& controller, Type type, Direction direction, u8 endpoint_address, u16 max_packet_size, i8 device_address, u8 poll_interval)
|
2021-02-27 15:10:37 +11:00
|
|
|
{
|
2021-08-12 04:48:54 +01:00
|
|
|
auto pipe = adopt_own_if_nonnull(new (nothrow) Pipe(controller, type, direction, endpoint_address, max_packet_size, poll_interval, device_address));
|
2021-02-27 15:10:37 +11:00
|
|
|
if (!pipe)
|
|
|
|
|
return ENOMEM;
|
|
|
|
|
|
|
|
|
|
return pipe.release_nonnull();
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-08 19:50:20 +01:00
|
|
|
Pipe::Pipe(USBController const& controller, Type type, Pipe::Direction direction, u16 max_packet_size)
|
|
|
|
|
: m_controller(controller)
|
|
|
|
|
, m_type(type)
|
2021-02-27 15:10:37 +11:00
|
|
|
, m_direction(direction)
|
|
|
|
|
, m_endpoint_address(0)
|
|
|
|
|
, m_max_packet_size(max_packet_size)
|
|
|
|
|
, m_poll_interval(0)
|
|
|
|
|
, m_data_toggle(false)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-08 19:50:20 +01:00
|
|
|
Pipe::Pipe(USBController const& controller, Type type, Direction direction, USBEndpointDescriptor& endpoint [[maybe_unused]])
|
|
|
|
|
: m_controller(controller)
|
|
|
|
|
, m_type(type)
|
2021-02-27 15:10:37 +11:00
|
|
|
, m_direction(direction)
|
|
|
|
|
{
|
|
|
|
|
// TODO: decode endpoint structure
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-08 19:50:20 +01:00
|
|
|
Pipe::Pipe(USBController const& controller, Type type, Direction direction, u8 endpoint_address, u16 max_packet_size, u8 poll_interval, i8 device_address)
|
|
|
|
|
: m_controller(controller)
|
|
|
|
|
, m_type(type)
|
2021-02-27 15:10:37 +11:00
|
|
|
, m_direction(direction)
|
|
|
|
|
, m_device_address(device_address)
|
|
|
|
|
, m_endpoint_address(endpoint_address)
|
|
|
|
|
, m_max_packet_size(max_packet_size)
|
|
|
|
|
, m_poll_interval(poll_interval)
|
|
|
|
|
, m_data_toggle(false)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
KResultOr<size_t> Pipe::control_transfer(u8 request_type, u8 request, u16 value, u16 index, u16 length, void* data)
|
|
|
|
|
{
|
|
|
|
|
USBRequestData usb_request;
|
|
|
|
|
|
|
|
|
|
usb_request.request_type = request_type;
|
|
|
|
|
usb_request.request = request;
|
|
|
|
|
usb_request.value = value;
|
|
|
|
|
usb_request.index = index;
|
|
|
|
|
usb_request.length = length;
|
|
|
|
|
|
|
|
|
|
auto transfer = Transfer::try_create(*this, length);
|
|
|
|
|
|
|
|
|
|
if (!transfer)
|
|
|
|
|
return ENOMEM;
|
|
|
|
|
|
|
|
|
|
transfer->set_setup_packet(usb_request);
|
|
|
|
|
|
2021-07-21 19:53:38 +02:00
|
|
|
dbgln_if(USB_DEBUG, "Pipe: Transfer allocated @ {}", transfer->buffer_physical());
|
2021-08-08 19:50:20 +01:00
|
|
|
auto transfer_len_or_error = m_controller->submit_control_transfer(*transfer);
|
2021-02-27 15:10:37 +11:00
|
|
|
|
|
|
|
|
if (transfer_len_or_error.is_error())
|
|
|
|
|
return transfer_len_or_error.error();
|
|
|
|
|
|
|
|
|
|
auto transfer_length = transfer_len_or_error.release_value();
|
|
|
|
|
|
|
|
|
|
// TODO: Check transfer for completion and copy data from transfer buffer into data
|
|
|
|
|
if (length > 0)
|
|
|
|
|
memcpy(reinterpret_cast<u8*>(data), transfer->buffer().as_ptr() + sizeof(USBRequestData), length);
|
|
|
|
|
|
|
|
|
|
dbgln_if(USB_DEBUG, "Pipe: Control Transfer complete!");
|
|
|
|
|
return transfer_length;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|